@dev-anywhere/proxy 0.2.2 → 0.2.4

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.
@@ -4,9 +4,14 @@
4
4
  import { spawn } from "child_process";
5
5
  import { fileURLToPath } from "url";
6
6
  var IS_DEV = false;
7
- function spawnScript(scriptBaseUrl, args = [], options = {}) {
7
+ function resolveTopLevelScript(name) {
8
+ const baseDir = IS_DEV ? new URL("../", import.meta.url) : new URL("./", import.meta.url);
9
+ return new URL(name, baseDir);
10
+ }
11
+ function spawnScript(script, args = [], options = {}) {
8
12
  const { unref = true, logger, ...rest } = options;
9
13
  const stdio = rest.stdio ?? (logger ? ["ignore", "ignore", "pipe"] : "ignore");
14
+ const scriptBaseUrl = typeof script === "string" ? resolveTopLevelScript(script) : script;
10
15
  const basePath = fileURLToPath(scriptBaseUrl);
11
16
  const scriptPath = `${basePath}${IS_DEV ? ".ts" : ".js"}`;
12
17
  const runtimeArgs = IS_DEV ? ["--import", "tsx", scriptPath, ...args] : [scriptPath, ...args];
@@ -52,4 +57,4 @@ function spawnScript(scriptBaseUrl, args = [], options = {}) {
52
57
  export {
53
58
  spawnScript
54
59
  };
55
- //# sourceMappingURL=chunk-ZUWAB67J.js.map
60
+ //# sourceMappingURL=chunk-IHUQNJF6.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/common/env.ts"],"sourcesContent":["import { spawn, type ChildProcess, type SpawnOptions } from \"node:child_process\";\nimport { fileURLToPath } from \"node:url\";\nimport type { Logger } from \"pino\";\n\n// 运行环境判断。tsup 打包时通过 define 把 process.env.NODE_ENV 替换为 \"production\",\n// IS_DEV 随之静态折叠为 false,dev 分支被 dead-code elimination 删除。\n// node --import tsx 跑源码时 NODE_ENV 通常未设置,默认 \"development\",IS_DEV 为 true。\nconst IS_DEV = (process.env.NODE_ENV ?? \"development\") !== \"production\";\n\ninterface SpawnScriptOptions extends Omit<SpawnOptions, \"detached\"> {\n // 默认 true。设为 false 时父进程退出会把子进程一并带走。\n unref?: boolean;\n // 传入后启用\"现场记录\":子进程 stderr 按行 pipe 进 logger.warn(stderr 原文语义模糊,\n // 可能只是 deprecation warning),非零退出和 spawn 失败进 logger.error(确定性失败)。\n // 需要自己精确控制 stderr 的调用方(例如 startDaemon 要按超时展示)不要在这里传 logger,\n // 改为显式 `stdio: [\"ignore\", \"ignore\", \"pipe\"]` 自行挂 handler。\n logger?: Logger;\n}\n\n/**\n * 顶层 entry(serve / session-worker)在 dev 下 `src/<name>.ts`、prod 下 `dist/<name>.js`。\n * env.ts 自身:dev 在 `src/common/env.ts`,prod 被 tsup 内联进 `dist/<chunk>.js`。\n * 因此从 env.ts 看,dev 走 `../<name>`、prod 走 `./<name>` 即可命中正确目录。\n *\n * 早期 API 让调用方在自己源文件里 `new URL(\"./serve\", import.meta.url)`,对顶层\n * caller(src/index.ts)巧合可用,但 src/terminal/、src/serve/ 这类嵌套 caller 在 src 下\n * 写 `../serve` 才对、tsup 把所有 chunk 拍平到 dist/ 根目录后又指错位置。把解析集中到\n * env.ts 这一处,调用方只传 entry 名,避免两层布局的相对路径陷阱。\n */\nexport function resolveTopLevelScript(name: string): URL {\n const baseDir = IS_DEV\n ? new URL(\"../\", import.meta.url) // src/common/env.ts → src/\n : new URL(\"./\", import.meta.url); // dist/<chunk>.js → dist/\n return new URL(name, baseDir);\n}\n\n/**\n * spawn 一个 Node 脚本作为后台 detached 子进程。\n *\n * 第一个参数支持两种形式:\n * - 顶层 entry 名(\"serve\" / \"session-worker\"):内部按 dev/prod 布局解析为\n * `src/<name>.ts` 或 `dist/<name>.js`。生产代码用这种形式。\n * - URL:fixture / 测试用,调用方自己用 `new URL(\"./xxx\", import.meta.url)` 构造。\n *\n * helper 根据 IS_DEV 自动补扩展名并选运行时:\n * dev: 执行 `node --import tsx <path>.ts`\n * prod: 执行 `node <path>.js`\n *\n * @example\n * spawnScript(\"serve\", [], { logger: terminalLogger });\n * spawnScript(\"session-worker\", [sessionId, sockPath], { logger });\n */\nexport function spawnScript(\n script: string | URL,\n args: readonly string[] = [],\n options: SpawnScriptOptions = {},\n): ChildProcess {\n const { unref = true, logger, ...rest } = options;\n // logger 传入时默认打开 stderr pipe;调用方显式传了 stdio 则尊重调用方。\n const stdio = rest.stdio ?? (logger ? [\"ignore\", \"ignore\", \"pipe\"] : \"ignore\");\n\n const scriptBaseUrl = typeof script === \"string\" ? resolveTopLevelScript(script) : script;\n const basePath = fileURLToPath(scriptBaseUrl);\n const scriptPath = `${basePath}${IS_DEV ? \".ts\" : \".js\"}`;\n const runtimeArgs = IS_DEV ? [\"--import\", \"tsx\", scriptPath, ...args] : [scriptPath, ...args];\n const child = spawn(process.execPath, runtimeArgs, {\n detached: true,\n ...rest,\n stdio,\n });\n\n if (logger) {\n if (child.stderr) {\n child.stderr.setEncoding(\"utf-8\");\n let buf = \"\";\n const emit = (line: string) => {\n const trimmed = line.trim();\n if (trimmed) logger.warn({ pid: child.pid, src: \"child-stderr\" }, trimmed);\n };\n child.stderr.on(\"data\", (chunk: string) => {\n buf += chunk;\n const lines = buf.split(\"\\n\");\n buf = lines.pop() ?? \"\";\n for (const line of lines) emit(line);\n });\n // 子进程结束时 buf 里可能还有不带 \\n 的残段(比如 stderr 最后一行无换行就退出),\n // 这里 flush 一次保证捕获 100% 行内容,不被\"没等到下一个 \\n\"吞掉。\n child.stderr.on(\"end\", () => {\n if (buf) {\n emit(buf);\n buf = \"\";\n }\n });\n }\n child.once(\"error\", (err) => {\n logger.error({ pid: child.pid, err: String(err) }, \"spawn failed\");\n });\n child.once(\"exit\", (code, signal) => {\n if (code !== 0 && code !== null) {\n logger.error({ pid: child.pid, code, signal }, \"child exited abnormally\");\n }\n });\n }\n\n if (unref) child.unref();\n return child;\n}\n"],"mappings":";;;AAAA,SAAS,aAAmD;AAC5D,SAAS,qBAAqB;AAM9B,IAAM,SAAU;AAsBT,SAAS,sBAAsB,MAAmB;AACvD,QAAM,UAAU,SACZ,IAAI,IAAI,OAAO,YAAY,GAAG,IAC9B,IAAI,IAAI,MAAM,YAAY,GAAG;AACjC,SAAO,IAAI,IAAI,MAAM,OAAO;AAC9B;AAkBO,SAAS,YACd,QACA,OAA0B,CAAC,GAC3B,UAA8B,CAAC,GACjB;AACd,QAAM,EAAE,QAAQ,MAAM,QAAQ,GAAG,KAAK,IAAI;AAE1C,QAAM,QAAQ,KAAK,UAAU,SAAS,CAAC,UAAU,UAAU,MAAM,IAAI;AAErE,QAAM,gBAAgB,OAAO,WAAW,WAAW,sBAAsB,MAAM,IAAI;AACnF,QAAM,WAAW,cAAc,aAAa;AAC5C,QAAM,aAAa,GAAG,QAAQ,GAAG,SAAS,QAAQ,KAAK;AACvD,QAAM,cAAc,SAAS,CAAC,YAAY,OAAO,YAAY,GAAG,IAAI,IAAI,CAAC,YAAY,GAAG,IAAI;AAC5F,QAAM,QAAQ,MAAM,QAAQ,UAAU,aAAa;AAAA,IACjD,UAAU;AAAA,IACV,GAAG;AAAA,IACH;AAAA,EACF,CAAC;AAED,MAAI,QAAQ;AACV,QAAI,MAAM,QAAQ;AAChB,YAAM,OAAO,YAAY,OAAO;AAChC,UAAI,MAAM;AACV,YAAM,OAAO,CAAC,SAAiB;AAC7B,cAAM,UAAU,KAAK,KAAK;AAC1B,YAAI,QAAS,QAAO,KAAK,EAAE,KAAK,MAAM,KAAK,KAAK,eAAe,GAAG,OAAO;AAAA,MAC3E;AACA,YAAM,OAAO,GAAG,QAAQ,CAAC,UAAkB;AACzC,eAAO;AACP,cAAM,QAAQ,IAAI,MAAM,IAAI;AAC5B,cAAM,MAAM,IAAI,KAAK;AACrB,mBAAW,QAAQ,MAAO,MAAK,IAAI;AAAA,MACrC,CAAC;AAGD,YAAM,OAAO,GAAG,OAAO,MAAM;AAC3B,YAAI,KAAK;AACP,eAAK,GAAG;AACR,gBAAM;AAAA,QACR;AAAA,MACF,CAAC;AAAA,IACH;AACA,UAAM,KAAK,SAAS,CAAC,QAAQ;AAC3B,aAAO,MAAM,EAAE,KAAK,MAAM,KAAK,KAAK,OAAO,GAAG,EAAE,GAAG,cAAc;AAAA,IACnE,CAAC;AACD,UAAM,KAAK,QAAQ,CAAC,MAAM,WAAW;AACnC,UAAI,SAAS,KAAK,SAAS,MAAM;AAC/B,eAAO,MAAM,EAAE,KAAK,MAAM,KAAK,MAAM,OAAO,GAAG,yBAAyB;AAAA,MAC1E;AAAA,IACF,CAAC;AAAA,EACH;AAEA,MAAI,MAAO,OAAM,MAAM;AACvB,SAAO;AACT;","names":[]}
@@ -1065,4 +1065,4 @@ export {
1065
1065
  serializeIpc,
1066
1066
  createIpcReader
1067
1067
  };
1068
- //# sourceMappingURL=chunk-NZZXBVO2.js.map
1068
+ //# sourceMappingURL=chunk-L6J5QCFH.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../packages/shared/src/schemas/envelope.ts","../../../packages/shared/src/schemas/id.ts","../../../packages/shared/src/schemas/chat.ts","../../../packages/shared/src/schemas/tool.ts","../../../packages/shared/src/schemas/session.ts","../../../packages/shared/src/constants/enums.ts","../../../packages/shared/src/constants/pty.ts","../../../packages/shared/src/schemas/system.ts","../../../packages/shared/src/builders/index.ts","../../../packages/shared/src/state-machine.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/binary-frame.ts","../src/ipc/ipc-protocol.ts","../src/ipc/line-buffer.ts"],"sourcesContent":["import { z } from \"zod\";\nimport { IdSchema } from \"./id.js\";\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// 信封基础字段:序列号、时间戳、来源、协议版本——所有 envelope 共用。\nconst BaseEnvelopeFields = {\n seq: z.number().int().nonnegative(),\n timestamp: z.number(),\n source: z.enum([\"proxy\", \"client\"]),\n version: z.string(),\n};\n\n// session-scoped envelope: 携带 sessionId 路由到具体会话, 大多数业务 envelope 走这里。\nconst SessionedEnvelopeFields = {\n ...BaseEnvelopeFields,\n sessionId: IdSchema,\n};\n\n// 按 type 字段区分的 discriminatedUnion 信封\nexport const MessageEnvelopeSchema = z.discriminatedUnion(\"type\", [\n // chat (3)\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"user_input\"),\n payload: UserInputPayloadSchema,\n }),\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"assistant_message\"),\n payload: AssistantMessagePayloadSchema,\n }),\n z.object({\n ...SessionedEnvelopeFields,\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 ...SessionedEnvelopeFields,\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 ...SessionedEnvelopeFields,\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 ...SessionedEnvelopeFields,\n type: z.literal(\"assistant_tool_use\"),\n payload: ToolUseRequestPayloadSchema,\n }),\n // session (5)\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"session_create\"),\n payload: SessionCreatePayloadSchema,\n }),\n // session_list 是全局广播 (列出所有 session), 不绑定具体 sessionId, 不携带该字段。\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_list\"),\n payload: SessionListPayloadSchema,\n }),\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"session_switch\"),\n payload: SessionSwitchPayloadSchema,\n }),\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"session_terminate\"),\n payload: SessionTerminatePayloadSchema,\n }),\n z.object({\n ...SessionedEnvelopeFields,\n type: z.literal(\"session_status\"),\n payload: SessionStatusPayloadSchema,\n }),\n // system (5): 心跳 / 认证 / 同步——全局, 无 sessionId\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// 所有跨进程传递的标识符 (proxyId / clientId / sessionId / requestId / toolId) 的统一上限。\n// 设计原因:\n// - nanoid (proxy/session/client 服务端默认生成) = 21 字符\n// - claude tool_use id (toolu_<base64ish>) ≈ 30-40 字符\n// - codex tool id 类似量级\n// 256 给协议演进留余量, 同时挡住 wire 上来的恶意超长 ID:\n// - Map<id, _> key 占用 (memory DoS)\n// - 日志行膨胀 (operator/磁盘压力)\n// - 拼到 disk path (虽然当前没有此路径, defense in depth)\nconst MAX_ID_LENGTH = 256;\n\nexport const IdSchema = z.string().min(1).max(MAX_ID_LENGTH);\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\";\nimport { IdSchema } from \"./id.js\";\n\n// 工具调用请求\nexport const ToolUseRequestPayloadSchema = z.object({\n toolName: z.string(),\n toolId: IdSchema,\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: IdSchema,\n whitelistTool: z.boolean().optional(),\n});\n\nexport type ToolApprovePayload = z.infer<typeof ToolApprovePayloadSchema>;\n\n// 工具调用拒绝\nexport const ToolDenyPayloadSchema = z.object({\n toolId: IdSchema,\n reason: z.string().optional(),\n});\n\nexport type ToolDenyPayload = z.infer<typeof ToolDenyPayloadSchema>;\n\n// 工具调用结果\nexport const ToolResultPayloadSchema = z.object({\n toolId: IdSchema,\n result: z.unknown(),\n isError: z.boolean(),\n});\n\nexport type ToolResultPayload = z.infer<typeof ToolResultPayloadSchema>;\n","import { z } from \"zod\";\nimport { providerValues, ptyOwnerValues, sessionModeValues } from \"../constants/enums.js\";\nimport { ptySemanticStateValues } from \"../constants/pty.js\";\nimport { IdSchema } from \"./id.js\";\n\nexport const sessionStateValues = [\n \"idle\",\n \"working\",\n \"waiting_approval\",\n \"error\",\n \"terminated\",\n] 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: IdSchema,\n name: z.string().optional(),\n state: z.enum(sessionStateValues),\n mode: z.enum(sessionModeValues).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: IdSchema,\n});\n\nexport type SessionSwitchPayload = z.infer<typeof SessionSwitchPayloadSchema>;\n\n// 终止会话\nexport const SessionTerminatePayloadSchema = z.object({\n sessionId: IdSchema,\n});\n\nexport type SessionTerminatePayload = z.infer<typeof SessionTerminatePayloadSchema>;\n\n// 会话状态变更\n// lastActive: 触发本次状态迁移或活动刷新的时间戳 (ms),用于列表相对时间显示。\nexport const SessionStatusPayloadSchema = z.object({\n sessionId: IdSchema,\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(ptySemanticStateValues),\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: IdSchema,\n toolName: z.string(),\n input: z.record(z.string(), z.unknown()),\n })\n .optional(),\n permissionResolution: z\n .object({\n requestId: IdSchema,\n outcome: z.enum([\"allow\", \"deny\"]),\n })\n .optional(),\n summary: z.string().optional(),\n});\nexport type AgentStatusPayload = z.infer<typeof AgentStatusPayloadSchema>;\n","// 协议层共用的枚举字面量集合。Zod schema、TypeScript 类型、运行时常量都从这里派生,\n// 避免在 relay-control / session / chat / web 各自手写 z.enum([\"claude\", \"codex\"]) 漂移。\n\nexport const providerValues = [\"claude\", \"codex\"] as const;\nexport type ProviderId = (typeof providerValues)[number];\n\nexport const ptyOwnerValues = [\"local-terminal\", \"proxy-hosted\"] as const;\nexport type PtyOwner = (typeof ptyOwnerValues)[number];\n\nexport const sessionModeValues = [\"pty\", \"json\"] as const;\nexport type SessionMode = (typeof sessionModeValues)[number];\n","// PTY 局部语义状态。仅承载明确语义信号; title/spinner 变化时事件的 state 取 null,\n// 让上层不参与 FSM 切换, 只走 title 推送通道。\n// 单一 source of truth: osc-extractor / ipc-protocol / shared schema 都从这里取。\nexport const PtySemanticState = {\n WORKING: \"working\",\n TURN_COMPLETE: \"turn_complete\",\n APPROVAL_WAIT: \"approval_wait\",\n} as const;\n\nexport type PtySemanticState = (typeof PtySemanticState)[keyof typeof PtySemanticState];\n\n// zod / 类型字面量场景下复用的 const tuple, z.enum 直接吃。\nexport const ptySemanticStateValues = [\n PtySemanticState.WORKING,\n PtySemanticState.TURN_COMPLETE,\n PtySemanticState.APPROVAL_WAIT,\n] as const;\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\";\nimport type { RelayControlMessage } from \"../schemas/relay-control.js\";\n\n// 构建经过 schema 验证的消息信封\n// seq 由调用方提供,必须与 EventStore per-session seq 一致,保证 proxy 和 relay 对账可靠\n// sessionId 仅 session-scoped envelope 携带; 全局广播 (session_list / heartbeat / auth /\n// sync_*) 传 null 或省略, 否则要求合法 ID。\nexport function buildMessage<T extends MessageEnvelope[\"type\"]>(\n type: T,\n sessionId: string | null,\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 !== null ? { 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\n// 序列化 relay control 消息。借助 RelayControlMessage 这个 discriminated union,\n// 调用方传入 { type, ... } 时 TypeScript 会按 type 反查出该消息允许携带的字段集,\n// 替代之前散落的 sendRaw(JSON.stringify({ type: \"...\", ... })) 模式 —— 后者在编译期\n// 完全没有约束,任何字段拼写错误 / 字段漏填都到 wire 上才暴露。\n//\n// 这里没有再做一次 RelayControlSchema.parse(msg):proxy 在热路径每条 envelope 都要发,\n// 重复 zod 解析对吞吐有压力;编译期类型检查已经能挡住大部分错误。\n// 真要做运行时校验时,调用 RelayControlSchema.parse(msg) 再 stringify 即可。\nexport function serializeControl(msg: RelayControlMessage): string {\n return JSON.stringify(msg);\n}\n","// 有限状态机 helper\n//\n// 提供显式转换表的小型 FSM,区分两类调用模式:\n// - transitionTo(throw):同步确定性流程里调用,非法转移代表 bug,立即暴露\n// - tryTransitionTo(bool):异步事件回调里调用,非法转移可能是吸收态残余事件,调用方按\n// isInAbsorbingState() 分级日志\n//\n// 吸收态采用传递闭包定义:终态(transitions=[])或所有出边都指向吸收态的状态都算吸收。\n// 这样 SessionState.ERROR (→[TERMINATED]) 和 RelayConnectionState.CLOSED ([]) 都被\n// 自动识别为吸收态,不用在 caller 里硬编码状态名。\n//\n// onTransition/onRejected 仅做日志/观测,不应 throw。\n\ninterface FSMDef<S extends string> {\n initial: S;\n // from-state → 允许转入的 to-state 列表;终态对应空数组\n transitions: Record<S, readonly S[]>;\n // 合法转换发生后触发,典型用法是结构化日志\n onTransition?: (from: S, to: S) => void;\n // tryTransitionTo 非法转移时触发;isAbsorbing 指示 from 是否吸收态(晚到残余 vs. 真非法)\n onRejected?: (from: S, to: S, isAbsorbing: boolean) => void;\n}\n\ninterface FSM<S extends string> {\n current(): S;\n is(state: S): boolean;\n isIn(states: readonly S[]): boolean;\n canTransitionTo(to: S): boolean;\n // 非法转换抛 Error;同步流程里当 assert 用\n transitionTo(to: S): void;\n // 非法转换返回 false,不抛;异步回调里用,配合 isInAbsorbingState 分级日志\n tryTransitionTo(to: S): boolean;\n // 当前是否在吸收态(传递闭包)\n isInAbsorbingState(): boolean;\n}\n\n// 计算吸收态集合:终态 + 所有出边都指向吸收态的状态,迭代至不动点\nfunction computeAbsorbingSet<S extends string>(transitions: Record<S, readonly S[]>): Set<S> {\n const absorbing = new Set<S>();\n const entries = Object.entries(transitions) as Array<[S, readonly S[]]>;\n for (const [s, outs] of entries) {\n if (outs.length === 0) absorbing.add(s);\n }\n let changed = true;\n while (changed) {\n changed = false;\n for (const [s, outs] of entries) {\n if (absorbing.has(s)) continue;\n if (outs.length > 0 && outs.every((t) => absorbing.has(t))) {\n absorbing.add(s);\n changed = true;\n }\n }\n }\n return absorbing;\n}\n\nexport function createFSM<S extends string>(def: FSMDef<S>): FSM<S> {\n let state = def.initial;\n const absorbing = computeAbsorbingSet(def.transitions);\n const tryTransitionTo = (to: S): boolean => {\n const allowed = def.transitions[state];\n if (!allowed?.includes(to)) {\n def.onRejected?.(state, to, absorbing.has(state));\n return false;\n }\n const from = state;\n state = to;\n def.onTransition?.(from, to);\n return true;\n };\n return {\n current: () => state,\n is: (s) => state === s,\n isIn: (ss) => ss.includes(state),\n canTransitionTo: (to) => def.transitions[state]?.includes(to) ?? false,\n transitionTo: (to) => {\n if (!tryTransitionTo(to)) {\n throw new Error(`Invalid FSM transition: ${state} -> ${to}`);\n }\n },\n tryTransitionTo,\n isInAbsorbingState: () => absorbing.has(state),\n };\n}\n\n// 无内部 state 的 FSM 视图,供 state 存在外部(如 SessionInfo、DB 行)的 per-instance 场景使用。\n// 调用方自行传入 from,canTransition 校验;吸收态判定通过 isAbsorbing(state) 提供。\ninterface StatelessFSM<S extends string> {\n canTransition(from: S, to: S): boolean;\n isAbsorbing(state: S): boolean;\n}\n\nexport function defineFSM<S extends string>(transitions: Record<S, readonly S[]>): StatelessFSM<S> {\n const absorbing = computeAbsorbingSet(transitions);\n return {\n canTransition: (from, to) => transitions[from]?.includes(to) ?? false,\n isAbsorbing: (state) => absorbing.has(state),\n };\n}\n","import { z } from \"zod\";\nimport { IdSchema } from \"./id.js\";\nimport { AgentStatusPayloadSchema, PtyStatePayloadSchema, sessionStateValues } from \"./session.js\";\nimport { ToolApprovePayloadSchema, ToolDenyPayloadSchema } from \"./tool.js\";\nimport { RelayErrorCode } from \"../constants/relay-errors.js\";\nimport { ControlErrorCode } from \"../constants/control-errors.js\";\nimport { providerValues, ptyOwnerValues, sessionModeValues } from \"../constants/enums.js\";\n\n// 控制消息中复用的子类型\nexport const ProxyInfoSchema = z.object({\n proxyId: IdSchema,\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(providerValues).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: IdSchema.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};\nconst ClipboardImageMimeTypeSchema = z.enum([\"image/png\", \"image/jpeg\", \"image/webp\", \"image/gif\"]);\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: IdSchema,\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: IdSchema }),\n control(\"proxy_select_response\", {\n ...RequestIdShape,\n success: z.boolean(),\n proxyId: IdSchema.optional(),\n ...RequestErrorShape,\n }),\n control(\"relay_error\", {\n code: z.enum(Object.values(RelayErrorCode) as [RelayErrorCode, ...RelayErrorCode[]]),\n message: z.string(),\n // 可选 requestId: relay 把 client 发来 raw 的 requestId 字段透传回来,\n // client 侧 waitForMessage 据此把对应 pending request 立即拒掉而不必等到 timeout。\n requestId: IdSchema.optional(),\n }),\n\n // 客户端注册协议\n control(\"client_register\", {\n clientId: IdSchema,\n }),\n control(\"client_register_response\", {\n status: z.enum([\"restored\", \"proxy_offline\", \"new\"]),\n proxyId: IdSchema.optional(),\n }),\n\n // Proxy 离线通知\n control(\"proxy_offline\", {\n proxyId: IdSchema,\n }),\n\n // Proxy 主动断开,relay 立即清理资源\n control(\"proxy_disconnect\", {\n proxyId: IdSchema,\n }),\n\n // Proxy 重连后通知 client 恢复\n control(\"proxy_online\", {\n proxyId: IdSchema,\n }),\n\n // 目录列表请求与响应\n control(\n \"dir_list_request\",\n {\n proxyId: IdSchema.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: IdSchema.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(\"pty_state\", { sessionId: IdSchema, payload: PtyStatePayloadSchema }, \"proxy_to_client\"),\n\n // Provider 语义状态,来自 Claude/Codex hook 等结构化事件,不从 PTY 字节推断\n control(\n \"agent_status\",\n { sessionId: IdSchema, payload: AgentStatusPayloadSchema },\n \"proxy_to_client\",\n ),\n\n // 终端标题变化,proxy -> client\n control(\"terminal_title\", { sessionId: IdSchema, title: z.string() }, \"proxy_to_client\"),\n\n // 终端尺寸变化,proxy -> client\n control(\n \"terminal_resize\",\n { sessionId: IdSchema, cols: z.number().int().positive(), rows: z.number().int().positive() },\n \"proxy_to_client\",\n ),\n control(\n \"terminal_resize_request\",\n { sessionId: IdSchema, 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: IdSchema }, \"client_to_proxy\"),\n\n // 中断当前 turn,client -> proxy,SIGINT 到 worker 进程让 claude CLI abort 当前流\n control(\"session_worker_abort\", { sessionId: IdSchema }, \"client_to_proxy\"),\n\n // turn 完成信号,proxy -> client,对应 claude stream-json 的 result 事件\n control(\n \"turn_result\",\n {\n sessionId: IdSchema,\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(\"remote_input_raw\", { sessionId: IdSchema, data: z.string() }, \"client_to_proxy\"),\n control(\n \"clipboard_image_upload\",\n {\n ...RequestIdShape,\n sessionId: IdSchema,\n mimeType: ClipboardImageMimeTypeSchema,\n dataBase64: z.string().min(1),\n fileName: z.string().optional(),\n },\n \"client_to_proxy\",\n ),\n control(\n \"clipboard_image_upload_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: IdSchema,\n success: z.boolean(),\n // success=false 时 proxy 没有有效 path 可填;保持 optional 以避免占位空字符串通过校验。\n path: z.string().optional(),\n },\n \"proxy_to_client\",\n ),\n control(\n \"image_preview_request\",\n {\n ...RequestIdShape,\n sessionId: IdSchema,\n path: z.string().min(1),\n },\n \"client_to_proxy\",\n ),\n control(\n \"image_preview_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: IdSchema,\n success: z.boolean(),\n // 同 clipboard_image_upload_response:失败时 proxy 不一定有路径。\n path: z.string().optional(),\n mimeType: ClipboardImageMimeTypeSchema.optional(),\n dataBase64: z.string().optional(),\n size: z.number().int().nonnegative().optional(),\n },\n \"proxy_to_client\",\n ),\n // 任意文件下载: 与 image_preview 形状对称, 只是 mimeType 不限定为图片;\n // 单租户场景下 path 任意 (不受 previewRoots 限制), 由 proxy 端 size cap 兜底。\n control(\n \"file_download_request\",\n {\n ...RequestIdShape,\n sessionId: IdSchema,\n path: z.string().min(1),\n },\n \"client_to_proxy\",\n ),\n control(\n \"file_download_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: IdSchema,\n success: z.boolean(),\n path: z.string().optional(),\n mimeType: z.string().optional(),\n dataBase64: z.string().optional(),\n size: z.number().int().nonnegative().optional(),\n },\n \"proxy_to_client\",\n ),\n // 任意文件上传: 复用 clipboard_image_upload 的形状, mimeType 放开 + fileName 必填,\n // 由 proxy 端写入 session cwd 的 .dev-anywhere/uploads/ 子目录, 返回相对路径供 web 拼成 @path。\n control(\n \"file_upload_request\",\n {\n ...RequestIdShape,\n sessionId: IdSchema,\n mimeType: z.string().min(1),\n dataBase64: z.string().min(1),\n fileName: z.string().min(1),\n },\n \"client_to_proxy\",\n ),\n control(\n \"file_upload_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: IdSchema,\n success: z.boolean(),\n path: z.string().optional(),\n },\n \"proxy_to_client\",\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(providerValues), 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(providerValues),\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(providerValues),\n mode: z.enum(sessionModeValues).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 // 失败路径只送 errorCode/error, sessionId 此时无语义。成功路径才有 id。\n sessionId: IdSchema.optional(),\n mode: z.enum(sessionModeValues).optional(),\n provider: z.enum(providerValues).optional(),\n ptyOwner: z.enum(ptyOwnerValues).optional(),\n ...RequestErrorShape,\n },\n \"proxy_to_client\",\n ),\n\n // 客户端请求会话历史消息,client -> proxy\n control(\n \"session_messages_request\",\n {\n ...RequestIdShape,\n sessionId: IdSchema,\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: IdSchema },\n \"client_to_proxy\",\n ),\n control(\n \"session_resources_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: IdSchema,\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: IdSchema.optional() },\n \"client_to_proxy\",\n ),\n control(\n \"agent_status_response\",\n {\n ...RequestIdShape,\n statuses: z.array(z.object({ sessionId: IdSchema, payload: AgentStatusPayloadSchema })),\n },\n \"proxy_to_client\",\n ),\n\n // 客户端确认已收到审批请求;proxy 只记录送达状态,不把它当成用户决策\n control(\n \"permission_request_delivered\",\n { sessionId: IdSchema, requestId: IdSchema },\n \"client_to_proxy\",\n ),\n control(\n \"tool_approve\",\n { sessionId: IdSchema, payload: ToolApprovePayloadSchema },\n \"client_to_proxy\",\n ),\n control(\"tool_deny\", { sessionId: IdSchema, payload: ToolDenyPayloadSchema }, \"client_to_proxy\"),\n\n // proxy 确认用户决策已进入 provider/worker 路径;web 用它更新审批卡片状态\n control(\n \"permission_decision_result\",\n {\n sessionId: IdSchema,\n requestId: IdSchema,\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: IdSchema,\n approvals: z.array(\n z.object({\n requestId: IdSchema,\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: IdSchema,\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。session_sync 由 relay 自消费(更新 proxy-session\n // 关联)不转发给 client,因此**没有** direction 标注——RelayControlDirection 只描述转发流。\n control(\"session_sync\", {\n sessions: z.array(\n z.object({\n id: z.string(),\n mode: z.enum(sessionModeValues),\n provider: z.enum(providerValues),\n ptyOwner: z.enum(ptyOwnerValues).optional(),\n state: z.enum(sessionStateValues),\n }),\n ),\n }),\n\n // PTY 会话订阅,client -> proxy,触发 terminal serialize() 返回当前状态\n control(\n \"session_subscribe\",\n { sessionId: IdSchema, requestId: IdSchema.optional() },\n \"client_to_proxy\",\n ),\n\n // PTY 会话快照,proxy -> client,serialize() 的全量终端状态\n control(\n \"session_snapshot\",\n {\n sessionId: IdSchema,\n cols: z.number().int().positive(),\n rows: z.number().int().positive(),\n data: z.string(),\n outputSeq: z.number().int().nonnegative(),\n requestId: IdSchema.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","// PTY binary 帧 wire 格式(跨 proxy / relay / web 三方通用):\n// [1B sessionId_len][sessionId UTF-8][4B outputSeq uint32LE][PTY data]\n// 任一方手写偏移量都极易在协议演进时漂移,统一走这里编/解码避免分叉。\n//\n// 用 Uint8Array 而非 Buffer:shared 包必须可以在浏览器构建里跑,\n// 不能依赖 Node-only 的 Buffer 全局。Node 调用方拿到的也是 Uint8Array\n// (Buffer 是 Uint8Array 的子类,反向兼容),传给 ws.send 等 API 没问题。\n\nexport interface DecodedBinaryFrame {\n sessionId: string;\n outputSeq: number;\n data: Uint8Array;\n}\n\nconst SID_LEN_BYTES = 1;\nconst SEQ_BYTES = 4;\nconst HEADER_FIXED_BYTES = SID_LEN_BYTES + SEQ_BYTES;\n\nexport function encodeBinaryFrame(\n sessionId: string,\n outputSeq: number,\n data: Uint8Array,\n): Uint8Array {\n const sidBytes = new TextEncoder().encode(sessionId);\n if (sidBytes.length === 0 || sidBytes.length > 255) {\n throw new RangeError(\n `sessionId byte length must be 1-255, got ${sidBytes.length} (sessionId=${sessionId})`,\n );\n }\n if (!Number.isInteger(outputSeq) || outputSeq < 0 || outputSeq > 0xffffffff) {\n throw new RangeError(`outputSeq must be a uint32, got ${outputSeq}`);\n }\n\n const frame = new Uint8Array(SID_LEN_BYTES + sidBytes.length + SEQ_BYTES + data.length);\n frame[0] = sidBytes.length;\n frame.set(sidBytes, SID_LEN_BYTES);\n const seqOffset = SID_LEN_BYTES + sidBytes.length;\n // little-endian 与 Node 端 Buffer.writeUInt32LE / 浏览器 DataView setUint32(_, _, true) 保持一致\n new DataView(frame.buffer, frame.byteOffset + seqOffset, SEQ_BYTES).setUint32(0, outputSeq, true);\n frame.set(data, seqOffset + SEQ_BYTES);\n return frame;\n}\n\n// 解码失败一律返回 null,调用方按业务诉求决定是丢帧还是关连接。\nexport function decodeBinaryFrame(view: Uint8Array): DecodedBinaryFrame | null {\n if (view.length < SID_LEN_BYTES + SEQ_BYTES) return null;\n const sidLen = view[0];\n if (sidLen === 0) return null;\n if (view.length < SID_LEN_BYTES + sidLen + SEQ_BYTES) return null;\n\n const sessionId = new TextDecoder().decode(view.subarray(SID_LEN_BYTES, SID_LEN_BYTES + sidLen));\n const seqOffset = SID_LEN_BYTES + sidLen;\n const outputSeq = new DataView(view.buffer, view.byteOffset + seqOffset, SEQ_BYTES).getUint32(\n 0,\n true,\n );\n const data = view.subarray(seqOffset + SEQ_BYTES);\n return { sessionId, outputSeq, data };\n}\n\n// 帧头长度(不含 payload data):调用方在做 chunk 拆分 / 估算缓冲时用得到。\nexport function binaryFrameHeaderLength(sessionId: string): number {\n const sidBytes = new TextEncoder().encode(sessionId);\n return HEADER_FIXED_BYTES + sidBytes.length;\n}\n","import { z } from \"zod\";\nimport {\n SessionState,\n encodeBinaryFrame,\n decodeBinaryFrame,\n ptySemanticStateValues,\n} 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// IPC binary 帧外层 = [1B marker][4B payload_len uint32LE] + 内层 PTY 帧(来自 shared/binary-frame)。\n// 内层格式([1B sid_len][sid][4B seq][data])由 encodeBinaryFrame 统一管理,\n// 避免与 hosted-pty-registry / terminal-ipc / web 各自手写偏移量分叉。\nexport function encodeBinaryIpcFrame(sessionId: string, data: Buffer, outputSeq: number): Buffer {\n const inner = encodeBinaryFrame(sessionId, outputSeq, data);\n const frame = Buffer.alloc(1 + 4 + inner.length);\n frame[0] = IPC_BINARY_MARKER;\n frame.writeUInt32LE(inner.length, 1);\n frame.set(inner, 5);\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 profile: z.string().optional(),\n relayName: z.string(),\n relayNameSource: z.enum([\"cli\", \"profile\"]),\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 z.object({\n type: z.literal(\"pty_semantic_event\"),\n sessionId: z.string(),\n state: z.enum(ptySemanticStateValues),\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\n// onProtocolError:单条 NDJSON 行 parse 失败 / schema 校验失败时回调(不会终止 reader)。\n// 历史上这里用 stream.emit(\"error\") 把传输层炸掉,触发 socket close 与 onDisconnect,等于让一条\n// 协议层的不兼容消息把整个 session 推进 ERROR 态——这是 Claude/Codex CLI 增加新事件类型时的真实风险。\n// 现在改为 callback:调用方自行决定 log + skip 还是断开,传输层保持开放。\nexport function createWorkerReader(\n stream: NodeJS.ReadableStream,\n onMessage: (msg: WorkerMessage) => void,\n onProtocolError?: (err: Error, line: string) => 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 onProtocolError?.(\n new Error(`Worker message validation failed: ${result.error.message}`),\n str,\n );\n }\n } catch (err) {\n onProtocolError?.(new Error(\"Worker message parse error\", { cause: err }), str);\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 重复解析每条消息。\n// 同 createWorkerReader:onProtocolError 让协议层 parse / schema 错误不再走 stream.emit(\"error\"),\n// 由调用方决定如何处理(warn-skip / disconnect)。默认 silent drop 是为了向后兼容尚未挂回调的调用点。\nexport function createIpcReader(\n stream: NodeJS.ReadableStream,\n onMessage: (msg: IpcMessage) => void,\n onBinaryFrame?: (sessionId: string, data: Buffer, outputSeq: number) => void,\n onProtocolError?: (err: Error, line: string) => 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],需要至少 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 内层就是 shared 端定义的 PTY frame([sid_len][sid][seq][data]),\n // 解码同样走 decodeBinaryFrame 保持单一权威。\n const decoded = decodeBinaryFrame(buf.subarray(5, totalFrameLen));\n if (decoded && onBinaryFrame) {\n // ptyData copy 保留与旧代码一致的语义:调用方拿到的是独立 Buffer,\n // 不会被后续 buf reslice 影响。\n onBinaryFrame(decoded.sessionId, Buffer.from(decoded.data), decoded.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 onProtocolError?.(\n new Error(`IPC message validation failed: ${result.error.message}`),\n line,\n );\n }\n } catch (err) {\n onProtocolError?.(new Error(\"IPC message parse error\", { cause: err }), line);\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\";\nimport { StringDecoder } from \"node:string_decoder\";\n\n// 将任意 data 事件分割为完整的 \\n 分隔行\n// Node.js data 事件不保证按行边界分割,此 Transform 保证每次 push 一个完整行\n//\n// 多字节 UTF-8 字符 (CJK 3 字节 / emoji 4 字节) 可能跨 chunk 边界。直接对单 chunk 调用\n// Buffer.toString() 会把不完整字节序列解码成 U+FFFD 替换字符——claude / codex CLI 输出\n// 中文 / emoji 时 stream-json 行变乱码, 解 JSON 失败被 schema 静默丢弃。StringDecoder\n// 跨 chunk 维护内部状态, 不完整字节缓存到下一次 write 拼上再解。\nexport class LineBuffer extends Transform {\n private buffer = \"\";\n private decoder = new StringDecoder(\"utf8\");\n\n _transform(chunk: Buffer | string, _encoding: BufferEncoding, callback: TransformCallback): void {\n this.buffer += typeof chunk === \"string\" ? chunk : this.decoder.write(chunk);\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 // decoder.end() 把仍在缓存里的不完整字节序列以 U+FFFD 形式吐出, 避免静默丢字节\n const tail = this.decoder.end();\n if (tail.length > 0) this.buffer += tail;\n if (this.buffer.length > 0) {\n this.push(this.buffer);\n this.buffer = \"\";\n }\n callback();\n }\n}\n"],"mappings":"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","names":["z","z","z","z","z","z","z","z","z","z","z","sessionStateValues","z"]}
@@ -101,4 +101,4 @@ export {
101
101
  extractOscSignals,
102
102
  decidePtySemanticTransition
103
103
  };
104
- //# sourceMappingURL=chunk-OBYEKZWC.js.map
104
+ //# sourceMappingURL=chunk-Q7AQFYHG.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/common/osc-extractor.ts","../src/common/pty-approval-state.ts","../src/common/pty-semantic-machine.ts"],"sourcesContent":["import type { PtySemanticState } from \"@dev-anywhere/shared\";\n\n// OSC 0: 窗口标题 -- ESC ] 0 ; <title> BEL/ST\n// OSC 9: 通知 -- ESC ] 9 ; <text> BEL/ST\n// 每次调用创建新的 regex 实例避免 g flag 导致的 lastIndex 状态泄漏\n// eslint-disable-next-line no-control-regex\nconst OSC_PATTERN = /\\x1b\\](\\d+);([^\\x07\\x1b]*?)(?:\\x07|\\x1b\\\\)/g;\n\ntype PtySignalProvider = \"claude\" | \"codex\";\n\ninterface PtyStateEvent {\n state: PtySemanticState | null;\n title?: string;\n tool?: string;\n}\n\ninterface OscSequence {\n code: number;\n text: string;\n}\n\nexport function extractOscSequences(rawData: string): OscSequence[] {\n const regex = new RegExp(OSC_PATTERN.source, OSC_PATTERN.flags);\n const matches: OscSequence[] = [];\n\n let match: RegExpExecArray | null;\n while ((match = regex.exec(rawData)) !== null) {\n matches.push({ code: parseInt(match[1], 10), text: match[2] });\n }\n\n return matches;\n}\n\nfunction lastSequence(matches: OscSequence[], code: number): OscSequence | undefined {\n for (let i = matches.length - 1; i >= 0; i -= 1) {\n if (matches[i].code === code) return matches[i];\n }\n return undefined;\n}\n\nfunction isCodexActionRequiredTitle(title: string): boolean {\n return /\\bAction Required\\b/i.test(title);\n}\n\n// 从 PTY 原始数据中提取 OSC 语义信号。\n// OSC 9 优先级高于 OSC 0,无匹配时返回 null。\n// 仅 OSC 0(spinner/标题)返回 { state: null, title },让调用方推 title 但不动 FSM。\nexport function extractOscSignals(\n rawData: string,\n provider?: PtySignalProvider,\n): PtyStateEvent | null {\n const matches = extractOscSequences(rawData);\n\n if (matches.length === 0) return null;\n\n const osc0 = lastSequence(matches, 0);\n\n // OSC 9 优先级更高,包含具体的语义信号;同帧 OSC 0 仍保留 title 给 UI。\n const osc9 = lastSequence(matches, 9);\n if (osc9) {\n if (osc9.text.includes(\"waiting for your input\") || osc9.text.trim() === \"4;0;\") {\n return { state: \"turn_complete\", ...(osc0 ? { title: osc0.text } : {}) };\n }\n if (osc9.text.includes(\"needs your permission\")) {\n const toolMatch = osc9.text.match(/permission.*?:\\s*(\\S+)/);\n return {\n state: \"approval_wait\",\n ...(osc0 ? { title: osc0.text } : {}),\n ...(toolMatch?.[1] ? { tool: toolMatch[1] } : {}),\n };\n }\n }\n\n if (provider === \"codex\" && osc0 && isCodexActionRequiredTitle(osc0.text)) {\n return { state: \"approval_wait\", title: osc0.text };\n }\n\n // 仅 OSC 0:标题/spinner 更新,没有明确语义信号。state=null 让上层只推 title。\n if (osc0 && !osc9) {\n return { state: null, title: osc0.text };\n }\n\n return null;\n}\n","import type { PtySemanticState } from \"@dev-anywhere/shared\";\n\n// signalState 三态:undefined 表示本帧没有任何 signal;null 表示有 signal 但只承载 title(OSC 0\n// only,无明确语义);其它值是明确语义状态。title-only signal 也视为对 approval_wait 的释放\n// 信号——codex 取消审批后 OSC 0 标题脱离 \"Action Required\" 即走这条路径。\nexport function shouldReleaseApprovalWait(options: {\n currentState: PtySemanticState;\n signalState: PtySemanticState | null | undefined;\n}): boolean {\n if (options.currentState !== \"approval_wait\") return false;\n if (options.signalState === undefined) return false;\n return options.signalState !== \"approval_wait\";\n}\n\nexport function stateAfterApprovalRelease(signalState: PtySemanticState | null): PtySemanticState {\n // title-only 释放(signal.state===null)= codex 取消审批语义,等用户下一轮输入。\n if (signalState === null) return \"turn_complete\";\n return signalState !== \"approval_wait\" ? signalState : \"working\";\n}\n","import type { PtySemanticState } from \"@dev-anywhere/shared\";\nimport { shouldReleaseApprovalWait, stateAfterApprovalRelease } from \"./pty-approval-state.js\";\n\n// PTY 语义状态机的纯决策层。从一段 PTY 输出抽到的 OSC 信号 + 当前 PTY 局部状态出发,\n// 决定下一态以及是否要 emit 一次 pty_state 事件。\n// 副作用(changeSessionState、onTurnComplete、IPC 写)由调用方负责,本模块零 IO,便于单测。\n//\n// signal.state 三态:明确语义(working/turn_complete/approval_wait)、null(仅 OSC 0 title\n// 更新无语义)、signal 整体为 null(本帧无 OSC)。title-only signal 在 approval_wait 上下文\n// 下视为释放信号(codex 取消审批),其它上下文不参与 FSM 决策。\n\ninterface PtySignal {\n state: PtySemanticState | null;\n title?: string;\n tool?: string;\n}\n\ninterface PtyTransitionInput {\n currentState: PtySemanticState;\n signal: PtySignal | null;\n // hosted-pty 在 JSON FSM 维度也维护 session.state;当外部 hook 已先把 session 推到\n // WAITING_APPROVAL 而本地 currentState 还停在 working 时,本帧仍应被解释为审批等待中。\n // terminal.ts 没有这个上下文,传 false / undefined 即可。\n sessionStateIsWaitingApproval?: boolean;\n}\n\ninterface PtyTransitionResult {\n // 决策后应当采用的 currentState 值;与 input.currentState 相同表示局部状态不变。\n nextState: PtySemanticState;\n // 是否要把这次决策作为一次 pty_state 事件外发(含 hosted 端的 sessionFsm 副作用)。\n emit: boolean;\n // 透传给 emit 事件的 meta;emit=false 时忽略。\n meta?: { title?: string; tool?: string };\n}\n\nfunction withMeta(signal: PtySignal | null | undefined): { title?: string; tool?: string } {\n return {\n ...(signal?.title !== undefined ? { title: signal.title } : {}),\n ...(signal?.tool !== undefined ? { tool: signal.tool } : {}),\n };\n}\n\nexport function decidePtySemanticTransition(input: PtyTransitionInput): PtyTransitionResult {\n const { currentState, signal, sessionStateIsWaitingApproval } = input;\n\n // 1. 显式 approval_wait 信号:进入 / 维持审批等待。\n if (signal?.state === \"approval_wait\") {\n return { nextState: \"approval_wait\", emit: true, meta: withMeta(signal) };\n }\n\n // 2. 已在 approval_wait 且收到非 approval_wait 信号(含 title-only):审批解除,\n // 落到 stateAfterApprovalRelease 决定的状态上。\n if (\n shouldReleaseApprovalWait({\n currentState,\n signalState: signal === null ? undefined : signal.state,\n })\n ) {\n return {\n nextState: stateAfterApprovalRelease(signal!.state),\n emit: true,\n meta: withMeta(signal),\n };\n }\n\n // 3. 审批上下文兜底:currentState=approval_wait 或 sessionState 已是 WAITING_APPROVAL 但还没收到\n // 解除信号。此时即便有其它信号也不应让 PTY 状态偏离审批等待,仍 re-emit approval_wait。\n // turn_complete 不在此列:它是合法的解除终点(由 #2 处理)。\n const inApprovalContext =\n currentState === \"approval_wait\" || sessionStateIsWaitingApproval === true;\n if (inApprovalContext && signal?.state !== \"turn_complete\") {\n return { nextState: \"approval_wait\", emit: true, meta: withMeta(signal) };\n }\n\n // 4. title-only signal(state=null,且不在审批上下文):不参与 FSM 决策,title 由调用方\n // 单独经 sendTerminalTitle 推。这里 emit=false 避免 spam pty_state。\n if (signal !== null && signal.state === null) {\n return { nextState: currentState, emit: false };\n }\n\n // 5. 任意非 working 显式信号(turn_complete):直接落到 signal.state。\n if (signal && signal.state !== null && signal.state !== \"working\") {\n return { nextState: signal.state, emit: true, meta: withMeta(signal) };\n }\n\n // 6. 隐式 working:当前不在 working 且本帧没有显式信号或信号 state=working,则推到 working。\n if (currentState !== \"working\") {\n return { nextState: \"working\", emit: true };\n }\n\n // 7. 无变化:current==working 且没有有效信号。不产生事件。\n return { nextState: currentState, emit: false };\n}\n"],"mappings":"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","names":[]}
package/dist/index.js CHANGED
@@ -5,11 +5,11 @@ import {
5
5
  } from "./chunk-3ZUZ22V6.js";
6
6
  import {
7
7
  spawnScript
8
- } from "./chunk-ZUWAB67J.js";
8
+ } from "./chunk-IHUQNJF6.js";
9
9
  import {
10
10
  createIpcReader,
11
11
  serializeIpc
12
- } from "./chunk-NZZXBVO2.js";
12
+ } from "./chunk-L6J5QCFH.js";
13
13
  import {
14
14
  CONFIG_PATH,
15
15
  PID_PATH,
@@ -191,7 +191,7 @@ async function startDaemon(options) {
191
191
  }
192
192
  if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);
193
193
  const serveArgs = ["--profile", PROFILE_NAME, ...daemonRelayArgs(options?.relayName)];
194
- const child = spawnScript(new URL("./serve", import.meta.url), serveArgs, {
194
+ const child = spawnScript("serve", serveArgs, {
195
195
  env: { ...process.env },
196
196
  stdio: ["ignore", "ignore", "pipe"],
197
197
  unref: false
@@ -246,7 +246,7 @@ var program = new Command("dev-anywhere").description("Dev Anywhere - transparen
246
246
  console.error(err instanceof Error ? err.message : String(err));
247
247
  process.exit(1);
248
248
  }
249
- const { startTerminal } = await import("./terminal-YGYIRO7H.js");
249
+ const { startTerminal } = await import("./terminal-THJIKZQG.js");
250
250
  const { provider, args } = invocation;
251
251
  await startTerminal(args, provider);
252
252
  });
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/cli-args.ts"],"sourcesContent":["import { existsSync, readFileSync, unlinkSync, writeFileSync } from \"node:fs\";\nimport { connect } from \"node:net\";\nimport { setTimeout as sleep } from \"node:timers/promises\";\nimport { fileURLToPath } from \"node:url\";\nimport { dirname, join } from \"node:path\";\nimport { Command } from \"commander\";\nimport {\n PID_PATH,\n SOCK_PATH,\n STOPPED_PATH,\n SERVICE_LOG_PATH,\n CONFIG_PATH,\n PROFILE_NAME,\n ensureProfileWorkspace,\n isInitialized,\n initWorkspace,\n} from \"./common/paths.js\";\nimport { spawnScript } from \"./common/env.js\";\nimport { daemonRelayArgs, setDesiredDaemonRelay } from \"./common/daemon-env.js\";\nimport { createIpcReader, serializeIpc } from \"./ipc/ipc-protocol.js\";\nimport { extractAgentInvocation, normalizeCliArgs, stripProxyProfileArgs } from \"./cli-args.js\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst pkg = JSON.parse(readFileSync(join(__dirname, \"..\", \"package.json\"), \"utf-8\")) as {\n version: string;\n};\n\nfunction stopService(): boolean {\n if (!existsSync(PID_PATH)) {\n console.error(\"Service is not running (no PID file)\");\n return false;\n }\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n try {\n process.kill(pid, \"SIGTERM\");\n console.log(`Service stopped (PID ${pid})`);\n } catch {\n console.error(`Process ${pid} not found, cleaning up stale files`);\n }\n if (existsSync(PID_PATH)) unlinkSync(PID_PATH);\n if (existsSync(SOCK_PATH)) unlinkSync(SOCK_PATH);\n writeFileSync(STOPPED_PATH, String(Date.now()));\n return true;\n}\n\nfunction showStatus(): Promise<number> {\n return new Promise((resolve) => {\n let lines = 0;\n const log = (s: string) => {\n console.log(s);\n lines++;\n };\n\n if (!existsSync(PID_PATH)) {\n log(`Profile: ${PROFILE_NAME}`);\n log(\"Service: not running\");\n resolve(lines);\n return;\n }\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n let alive = false;\n try {\n process.kill(pid, 0);\n alive = true;\n } catch {\n // process.kill(pid, 0) 抛错表示进程不存在\n }\n\n if (!alive) {\n log(\"Service: dead (stale PID file)\");\n resolve(lines);\n return;\n }\n\n log(`Profile: ${PROFILE_NAME}`);\n log(`Service: running (PID ${pid})`);\n log(`Socket: ${SOCK_PATH}`);\n log(`Log: ${SERVICE_LOG_PATH}`);\n\n const sock = connect(SOCK_PATH);\n sock.on(\"error\", () => {\n log(\"Sessions: unable to connect\");\n sock.destroy();\n resolve(lines);\n });\n sock.on(\"connect\", () => {\n createIpcReader(sock, (msg) => {\n if (msg.type === \"service_status_response\") {\n const config = msg.config;\n log(`Daemon: profile ${config.profile ?? PROFILE_NAME}`);\n log(`Relay: ${config.relayName} (${config.relayNameSource})`);\n log(`Config: relay ${config.relayUrl ?? \"(unset)\"} (${config.relayUrlSource})`);\n const relay = msg.relay;\n if (!relay) {\n log(\"Relay: not configured\");\n } else if (relay.connected) {\n log(`Relay: connected (proxy: ${relay.proxyId})`);\n log(\n ` queue depth: ${relay.queueDepth}, reconnect attempts: ${relay.reconnectAttempt}`,\n );\n } else {\n log(\n `Relay: disconnected (proxy: ${relay.proxyId}, reconnecting: attempt ${relay.reconnectAttempt}, queued: ${relay.queueDepth})`,\n );\n }\n log(\"\");\n\n // 显示会话列表\n const sessions = msg.sessions;\n if (sessions.length === 0) {\n log(\"Sessions: none\");\n } else {\n log(`Sessions: ${sessions.length}`);\n for (const s of sessions) {\n log(` ${s.id} ${s.mode} ${s.state} worker: ${s.hasWorker ? \"yes\" : \"no\"}`);\n }\n }\n sock.destroy();\n resolve(lines);\n }\n });\n sock.write(serializeIpc({ type: \"service_status_request\" }));\n });\n });\n}\n\nconst DAEMON_STARTUP_TIMEOUT_MS = 30_000;\nconst DAEMON_STARTUP_POLL_MS = 200;\n\n// 轮询 SOCK_PATH 直到可连接,作为 serve 的 readiness 信号。\n// serve.ts 里 server.listen(SOCK_PATH) 是启动序列的最后一步,连上即代表 ready。\nasync function waitForServeReady(timeoutMs: number): Promise<boolean> {\n const deadline = Date.now() + timeoutMs;\n while (Date.now() < deadline) {\n const connected = await new Promise<boolean>((resolve) => {\n const sock = connect(SOCK_PATH);\n sock.once(\"connect\", () => {\n sock.destroy();\n resolve(true);\n });\n sock.once(\"error\", () => resolve(false));\n });\n if (connected) return true;\n await sleep(DAEMON_STARTUP_POLL_MS);\n }\n return false;\n}\n\nasync function startDaemon(options?: { relayName?: string }): Promise<void> {\n ensureProfileWorkspace();\n if (existsSync(PID_PATH)) {\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n try {\n process.kill(pid, 0);\n console.error(`Service is already running (PID ${pid})`);\n return;\n } catch {\n // process.kill(pid, 0) 抛错表示进程不存在,继续启动\n }\n }\n if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);\n\n // stderr 走 pipe 由父 CLI 订阅:子进程 ready 前(pino logger 未接管)的启动错误\n // 会被捕获;ready 后父 detach,pino 接管所有输出到 service.log。\n // start 命令必须等 daemon socket 可连接后再退出;否则用户会看到“启动成功”,实际服务还没就绪。\n const serveArgs = [\"--profile\", PROFILE_NAME, ...daemonRelayArgs(options?.relayName)];\n const child = spawnScript(new URL(\"./serve\", import.meta.url), serveArgs, {\n env: { ...process.env },\n stdio: [\"ignore\", \"ignore\", \"pipe\"],\n unref: false,\n });\n\n const stderrChunks: Buffer[] = [];\n child.stderr!.on(\"data\", (chunk: Buffer) => {\n stderrChunks.push(chunk);\n });\n\n // race: readiness handshake vs. 子进程先挂。子进程 ready 前就 exit 说明启动硬失败,\n // 不必再等到 30s 超时才报错。\n type Outcome =\n | { kind: \"ready\" }\n | { kind: \"timeout\" }\n | { kind: \"exited\"; code: number | null; signal: NodeJS.Signals | null };\n\n const readyOutcome: Promise<Outcome> = waitForServeReady(DAEMON_STARTUP_TIMEOUT_MS).then((ok) =>\n ok ? { kind: \"ready\" as const } : { kind: \"timeout\" as const },\n );\n const exitOutcome: Promise<Outcome> = new Promise((resolve) => {\n // 设 listener 前已经 exit 的边界:Node 记在 exitCode 上\n if (child.exitCode !== null) {\n resolve({ kind: \"exited\", code: child.exitCode, signal: child.signalCode });\n return;\n }\n child.once(\"exit\", (code, signal) => resolve({ kind: \"exited\", code, signal }));\n });\n\n const result = await Promise.race([readyOutcome, exitOutcome]);\n\n if (result.kind === \"ready\") {\n console.log(`Service started in background (PID ${child.pid})`);\n // ready 后 detach:摘 stderr 订阅 + destroy pipe + unref 子进程。\n // 单独 child.unref() 不够,父侧的 stderr pipe fd 还在事件循环里会让父 CLI 永不退出;\n // 必须 destroy 掉 pipe 才能真正释放 refcount。pino 已接管子进程的输出到 service.log。\n child.stderr!.removeAllListeners(\"data\");\n child.stderr!.destroy();\n child.unref();\n return;\n }\n\n // 失败路径:timeout 或 exited\n const stderrOutput = Buffer.concat(stderrChunks).toString(\"utf-8\").trim();\n if (result.kind === \"exited\") {\n console.error(`Service exited during startup (code=${result.code}, signal=${result.signal}).`);\n } else {\n console.error(`Service failed to become ready within ${DAEMON_STARTUP_TIMEOUT_MS / 1000}s.`);\n try {\n process.kill(child.pid!, \"SIGTERM\");\n } catch {\n // 子进程可能已自己退出,kill 失败不影响后续退出码\n }\n }\n if (stderrOutput) {\n console.error(\"--- child stderr ---\");\n console.error(stderrOutput);\n }\n process.exit(1);\n}\n\nconst program = new Command(\"dev-anywhere\")\n .description(\"Dev Anywhere - transparent local AI CLI proxy with remote control\")\n .version(pkg.version, \"-v, --version\")\n .option(\"--profile <name>\", \"Use an isolated local proxy profile\")\n .allowUnknownOption()\n .allowExcessArguments()\n .action(async () => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n // 参数校验放在 dynamic import 之前:错误参数路径不应触发 terminal 模块加载,\n // 避免无谓地拉起 PTY/xterm/logger 这些重资源(也避免 logger 文件 IO 副作用)。\n let invocation: ReturnType<typeof extractAgentInvocation>;\n try {\n invocation = extractAgentInvocation(cliArgsWithoutProfile);\n } catch (err) {\n console.error(err instanceof Error ? err.message : String(err));\n process.exit(1);\n }\n // 延迟导入 terminal: CLI 的其他子命令(init/stop/status)不需要 PTY + xterm 相关依赖,\n // tsup 基于 dynamic import 自动代码分裂,避免所有命令都为 terminal 付出 14KB 额外启动成本。\n const { startTerminal } = await import(\"./terminal.js\");\n const { provider, args } = invocation;\n await startTerminal(args, provider);\n });\n\n// serve 子命令组\nconst serve = new Command(\"serve\")\n .description(\"Manage the dev-anywhere background service\")\n .option(\"--profile <name>\", \"Use an isolated local proxy profile\")\n .option(\"-d, --daemon\", \"Run in background\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n if (opts.daemon) {\n setDesiredDaemonRelay(undefined);\n await startDaemon();\n } else {\n // 延迟导入 serve: daemon 模式只需要 startDaemon(纯 spawn),不需要加载 70KB 的 serve bundle\n const { startService } = await import(\"./serve.js\");\n await startService();\n }\n });\n\nserve\n .command(\"start\")\n .description(\"Start the background service\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n setDesiredDaemonRelay(opts.relay);\n await startDaemon({ relayName: opts.relay });\n });\n\nserve\n .command(\"status\")\n .description(\"Show service status and active sessions\")\n .option(\"-w, --watch\", \"Continuous monitoring mode\")\n .option(\"-n, --interval <seconds>\", \"Refresh interval in seconds\", \"2\")\n .action(async (opts) => {\n if (opts.watch) {\n const intervalMs = Number(opts.interval) * 1000;\n let lastLines = await showStatus();\n setInterval(async () => {\n if (lastLines > 0) {\n process.stdout.write(`\\x1B[${lastLines}A\\x1B[J`);\n }\n lastLines = await showStatus();\n }, intervalMs);\n } else {\n await showStatus();\n }\n });\n\nserve\n .command(\"stop\")\n .description(\"Stop the background service\")\n .action(() => {\n stopService();\n });\n\nserve\n .command(\"restart\")\n .description(\"Restart the background service\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n setDesiredDaemonRelay(opts.relay);\n stopService();\n await startDaemon({ relayName: opts.relay });\n });\n\nprogram.addCommand(serve);\n\nconst relay = new Command(\"relay\").description(\"Inspect and manage relay configuration\");\n\nrelay\n .command(\"token\")\n .description(\"Print the relay's current client token (auth: proxy token)\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n const { runRelayTokenCommand } = await import(\"./relay-token.js\");\n await runRelayTokenCommand({ relayName: opts.relay });\n });\n\nprogram.addCommand(relay);\n\nprogram\n .command(\"init\")\n .description(\"Initialize dev-anywhere workspace (~/.dev-anywhere)\")\n .action(() => {\n if (isInitialized()) {\n console.log(`Already initialized. Config at ${CONFIG_PATH}`);\n return;\n }\n initWorkspace();\n console.log(\"Initialized ~/.dev-anywhere/\");\n console.log(`Edit ${CONFIG_PATH} to configure relay server URL.`);\n });\n\n// pnpm run dev -- args 会在参数前插入 \"--\"。根脚本和用户命令都可能再加一层\n// 分隔符,所以这里过滤所有前导分隔符,再交给 Commander 和 provider 参数解析。\nconst cliArgs = normalizeCliArgs(process.argv.slice(2));\nconst cliArgsWithoutProfile = stripProxyProfileArgs(cliArgs);\n\nprogram.parse(cliArgsWithoutProfile, { from: \"user\" });\n","import type { ProviderId } from \"./providers/index.js\";\n\nexport function normalizeCliArgs(args: string[]): string[] {\n const normalized = [...args];\n while (normalized[0] === \"--\") {\n normalized.shift();\n }\n return normalized;\n}\n\nexport function stripProxyProfileArgs(args: string[]): string[] {\n const result: string[] = [];\n for (let i = 0; i < args.length; i++) {\n const arg = args[i];\n if (arg === \"claude\" || arg === \"codex\") {\n result.push(...args.slice(i));\n break;\n }\n if (arg === \"--profile\") {\n i++;\n continue;\n }\n if (arg.startsWith(\"--profile=\")) {\n continue;\n }\n result.push(arg);\n }\n return result;\n}\n\nexport function extractAgentInvocation(args: string[]): { provider: ProviderId; args: string[] } {\n const [agent, ...providerArgs] = args;\n if (agent !== \"claude\" && agent !== \"codex\") {\n throw new Error(\n 'Missing Agent CLI. Use \"dev-anywhere claude ...\" or \"dev-anywhere codex ...\".',\n );\n }\n return { provider: agent, args: providerArgs };\n}\n"],"mappings":"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","names":["relay"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/cli-args.ts"],"sourcesContent":["import { existsSync, readFileSync, unlinkSync, writeFileSync } from \"node:fs\";\nimport { connect } from \"node:net\";\nimport { setTimeout as sleep } from \"node:timers/promises\";\nimport { fileURLToPath } from \"node:url\";\nimport { dirname, join } from \"node:path\";\nimport { Command } from \"commander\";\nimport {\n PID_PATH,\n SOCK_PATH,\n STOPPED_PATH,\n SERVICE_LOG_PATH,\n CONFIG_PATH,\n PROFILE_NAME,\n ensureProfileWorkspace,\n isInitialized,\n initWorkspace,\n} from \"./common/paths.js\";\nimport { spawnScript } from \"./common/env.js\";\nimport { daemonRelayArgs, setDesiredDaemonRelay } from \"./common/daemon-env.js\";\nimport { createIpcReader, serializeIpc } from \"./ipc/ipc-protocol.js\";\nimport { extractAgentInvocation, normalizeCliArgs, stripProxyProfileArgs } from \"./cli-args.js\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst pkg = JSON.parse(readFileSync(join(__dirname, \"..\", \"package.json\"), \"utf-8\")) as {\n version: string;\n};\n\nfunction stopService(): boolean {\n if (!existsSync(PID_PATH)) {\n console.error(\"Service is not running (no PID file)\");\n return false;\n }\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n try {\n process.kill(pid, \"SIGTERM\");\n console.log(`Service stopped (PID ${pid})`);\n } catch {\n console.error(`Process ${pid} not found, cleaning up stale files`);\n }\n if (existsSync(PID_PATH)) unlinkSync(PID_PATH);\n if (existsSync(SOCK_PATH)) unlinkSync(SOCK_PATH);\n writeFileSync(STOPPED_PATH, String(Date.now()));\n return true;\n}\n\nfunction showStatus(): Promise<number> {\n return new Promise((resolve) => {\n let lines = 0;\n const log = (s: string) => {\n console.log(s);\n lines++;\n };\n\n if (!existsSync(PID_PATH)) {\n log(`Profile: ${PROFILE_NAME}`);\n log(\"Service: not running\");\n resolve(lines);\n return;\n }\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n let alive = false;\n try {\n process.kill(pid, 0);\n alive = true;\n } catch {\n // process.kill(pid, 0) 抛错表示进程不存在\n }\n\n if (!alive) {\n log(\"Service: dead (stale PID file)\");\n resolve(lines);\n return;\n }\n\n log(`Profile: ${PROFILE_NAME}`);\n log(`Service: running (PID ${pid})`);\n log(`Socket: ${SOCK_PATH}`);\n log(`Log: ${SERVICE_LOG_PATH}`);\n\n const sock = connect(SOCK_PATH);\n sock.on(\"error\", () => {\n log(\"Sessions: unable to connect\");\n sock.destroy();\n resolve(lines);\n });\n sock.on(\"connect\", () => {\n createIpcReader(sock, (msg) => {\n if (msg.type === \"service_status_response\") {\n const config = msg.config;\n log(`Daemon: profile ${config.profile ?? PROFILE_NAME}`);\n log(`Relay: ${config.relayName} (${config.relayNameSource})`);\n log(`Config: relay ${config.relayUrl ?? \"(unset)\"} (${config.relayUrlSource})`);\n const relay = msg.relay;\n if (!relay) {\n log(\"Relay: not configured\");\n } else if (relay.connected) {\n log(`Relay: connected (proxy: ${relay.proxyId})`);\n log(\n ` queue depth: ${relay.queueDepth}, reconnect attempts: ${relay.reconnectAttempt}`,\n );\n } else {\n log(\n `Relay: disconnected (proxy: ${relay.proxyId}, reconnecting: attempt ${relay.reconnectAttempt}, queued: ${relay.queueDepth})`,\n );\n }\n log(\"\");\n\n // 显示会话列表\n const sessions = msg.sessions;\n if (sessions.length === 0) {\n log(\"Sessions: none\");\n } else {\n log(`Sessions: ${sessions.length}`);\n for (const s of sessions) {\n log(` ${s.id} ${s.mode} ${s.state} worker: ${s.hasWorker ? \"yes\" : \"no\"}`);\n }\n }\n sock.destroy();\n resolve(lines);\n }\n });\n sock.write(serializeIpc({ type: \"service_status_request\" }));\n });\n });\n}\n\nconst DAEMON_STARTUP_TIMEOUT_MS = 30_000;\nconst DAEMON_STARTUP_POLL_MS = 200;\n\n// 轮询 SOCK_PATH 直到可连接,作为 serve 的 readiness 信号。\n// serve.ts 里 server.listen(SOCK_PATH) 是启动序列的最后一步,连上即代表 ready。\nasync function waitForServeReady(timeoutMs: number): Promise<boolean> {\n const deadline = Date.now() + timeoutMs;\n while (Date.now() < deadline) {\n const connected = await new Promise<boolean>((resolve) => {\n const sock = connect(SOCK_PATH);\n sock.once(\"connect\", () => {\n sock.destroy();\n resolve(true);\n });\n sock.once(\"error\", () => resolve(false));\n });\n if (connected) return true;\n await sleep(DAEMON_STARTUP_POLL_MS);\n }\n return false;\n}\n\nasync function startDaemon(options?: { relayName?: string }): Promise<void> {\n ensureProfileWorkspace();\n if (existsSync(PID_PATH)) {\n const pid = parseInt(readFileSync(PID_PATH, \"utf-8\").trim(), 10);\n try {\n process.kill(pid, 0);\n console.error(`Service is already running (PID ${pid})`);\n return;\n } catch {\n // process.kill(pid, 0) 抛错表示进程不存在,继续启动\n }\n }\n if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);\n\n // stderr 走 pipe 由父 CLI 订阅:子进程 ready 前(pino logger 未接管)的启动错误\n // 会被捕获;ready 后父 detach,pino 接管所有输出到 service.log。\n // start 命令必须等 daemon socket 可连接后再退出;否则用户会看到“启动成功”,实际服务还没就绪。\n const serveArgs = [\"--profile\", PROFILE_NAME, ...daemonRelayArgs(options?.relayName)];\n const child = spawnScript(\"serve\", serveArgs, {\n env: { ...process.env },\n stdio: [\"ignore\", \"ignore\", \"pipe\"],\n unref: false,\n });\n\n const stderrChunks: Buffer[] = [];\n child.stderr!.on(\"data\", (chunk: Buffer) => {\n stderrChunks.push(chunk);\n });\n\n // race: readiness handshake vs. 子进程先挂。子进程 ready 前就 exit 说明启动硬失败,\n // 不必再等到 30s 超时才报错。\n type Outcome =\n | { kind: \"ready\" }\n | { kind: \"timeout\" }\n | { kind: \"exited\"; code: number | null; signal: NodeJS.Signals | null };\n\n const readyOutcome: Promise<Outcome> = waitForServeReady(DAEMON_STARTUP_TIMEOUT_MS).then((ok) =>\n ok ? { kind: \"ready\" as const } : { kind: \"timeout\" as const },\n );\n const exitOutcome: Promise<Outcome> = new Promise((resolve) => {\n // 设 listener 前已经 exit 的边界:Node 记在 exitCode 上\n if (child.exitCode !== null) {\n resolve({ kind: \"exited\", code: child.exitCode, signal: child.signalCode });\n return;\n }\n child.once(\"exit\", (code, signal) => resolve({ kind: \"exited\", code, signal }));\n });\n\n const result = await Promise.race([readyOutcome, exitOutcome]);\n\n if (result.kind === \"ready\") {\n console.log(`Service started in background (PID ${child.pid})`);\n // ready 后 detach:摘 stderr 订阅 + destroy pipe + unref 子进程。\n // 单独 child.unref() 不够,父侧的 stderr pipe fd 还在事件循环里会让父 CLI 永不退出;\n // 必须 destroy 掉 pipe 才能真正释放 refcount。pino 已接管子进程的输出到 service.log。\n child.stderr!.removeAllListeners(\"data\");\n child.stderr!.destroy();\n child.unref();\n return;\n }\n\n // 失败路径:timeout 或 exited\n const stderrOutput = Buffer.concat(stderrChunks).toString(\"utf-8\").trim();\n if (result.kind === \"exited\") {\n console.error(`Service exited during startup (code=${result.code}, signal=${result.signal}).`);\n } else {\n console.error(`Service failed to become ready within ${DAEMON_STARTUP_TIMEOUT_MS / 1000}s.`);\n try {\n process.kill(child.pid!, \"SIGTERM\");\n } catch {\n // 子进程可能已自己退出,kill 失败不影响后续退出码\n }\n }\n if (stderrOutput) {\n console.error(\"--- child stderr ---\");\n console.error(stderrOutput);\n }\n process.exit(1);\n}\n\nconst program = new Command(\"dev-anywhere\")\n .description(\"Dev Anywhere - transparent local AI CLI proxy with remote control\")\n .version(pkg.version, \"-v, --version\")\n .option(\"--profile <name>\", \"Use an isolated local proxy profile\")\n .allowUnknownOption()\n .allowExcessArguments()\n .action(async () => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n // 参数校验放在 dynamic import 之前:错误参数路径不应触发 terminal 模块加载,\n // 避免无谓地拉起 PTY/xterm/logger 这些重资源(也避免 logger 文件 IO 副作用)。\n let invocation: ReturnType<typeof extractAgentInvocation>;\n try {\n invocation = extractAgentInvocation(cliArgsWithoutProfile);\n } catch (err) {\n console.error(err instanceof Error ? err.message : String(err));\n process.exit(1);\n }\n // 延迟导入 terminal: CLI 的其他子命令(init/stop/status)不需要 PTY + xterm 相关依赖,\n // tsup 基于 dynamic import 自动代码分裂,避免所有命令都为 terminal 付出 14KB 额外启动成本。\n const { startTerminal } = await import(\"./terminal.js\");\n const { provider, args } = invocation;\n await startTerminal(args, provider);\n });\n\n// serve 子命令组\nconst serve = new Command(\"serve\")\n .description(\"Manage the dev-anywhere background service\")\n .option(\"--profile <name>\", \"Use an isolated local proxy profile\")\n .option(\"-d, --daemon\", \"Run in background\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n if (opts.daemon) {\n setDesiredDaemonRelay(undefined);\n await startDaemon();\n } else {\n // 延迟导入 serve: daemon 模式只需要 startDaemon(纯 spawn),不需要加载 70KB 的 serve bundle\n const { startService } = await import(\"./serve.js\");\n await startService();\n }\n });\n\nserve\n .command(\"start\")\n .description(\"Start the background service\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n setDesiredDaemonRelay(opts.relay);\n await startDaemon({ relayName: opts.relay });\n });\n\nserve\n .command(\"status\")\n .description(\"Show service status and active sessions\")\n .option(\"-w, --watch\", \"Continuous monitoring mode\")\n .option(\"-n, --interval <seconds>\", \"Refresh interval in seconds\", \"2\")\n .action(async (opts) => {\n if (opts.watch) {\n const intervalMs = Number(opts.interval) * 1000;\n let lastLines = await showStatus();\n setInterval(async () => {\n if (lastLines > 0) {\n process.stdout.write(`\\x1B[${lastLines}A\\x1B[J`);\n }\n lastLines = await showStatus();\n }, intervalMs);\n } else {\n await showStatus();\n }\n });\n\nserve\n .command(\"stop\")\n .description(\"Stop the background service\")\n .action(() => {\n stopService();\n });\n\nserve\n .command(\"restart\")\n .description(\"Restart the background service\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n setDesiredDaemonRelay(opts.relay);\n stopService();\n await startDaemon({ relayName: opts.relay });\n });\n\nprogram.addCommand(serve);\n\nconst relay = new Command(\"relay\").description(\"Inspect and manage relay configuration\");\n\nrelay\n .command(\"token\")\n .description(\"Print the relay's current client token (auth: proxy token)\")\n .option(\"--relay <name>\", \"Use a named relay from config\")\n .action(async (opts) => {\n if (!isInitialized()) {\n console.error(`Dev Anywhere is not initialized. Run \"dev-anywhere init\" first.`);\n process.exit(1);\n }\n const { runRelayTokenCommand } = await import(\"./relay-token.js\");\n await runRelayTokenCommand({ relayName: opts.relay });\n });\n\nprogram.addCommand(relay);\n\nprogram\n .command(\"init\")\n .description(\"Initialize dev-anywhere workspace (~/.dev-anywhere)\")\n .action(() => {\n if (isInitialized()) {\n console.log(`Already initialized. Config at ${CONFIG_PATH}`);\n return;\n }\n initWorkspace();\n console.log(\"Initialized ~/.dev-anywhere/\");\n console.log(`Edit ${CONFIG_PATH} to configure relay server URL.`);\n });\n\n// pnpm run dev -- args 会在参数前插入 \"--\"。根脚本和用户命令都可能再加一层\n// 分隔符,所以这里过滤所有前导分隔符,再交给 Commander 和 provider 参数解析。\nconst cliArgs = normalizeCliArgs(process.argv.slice(2));\nconst cliArgsWithoutProfile = stripProxyProfileArgs(cliArgs);\n\nprogram.parse(cliArgsWithoutProfile, { from: \"user\" });\n","import type { ProviderId } from \"./providers/index.js\";\n\nexport function normalizeCliArgs(args: string[]): string[] {\n const normalized = [...args];\n while (normalized[0] === \"--\") {\n normalized.shift();\n }\n return normalized;\n}\n\nexport function stripProxyProfileArgs(args: string[]): string[] {\n const result: string[] = [];\n for (let i = 0; i < args.length; i++) {\n const arg = args[i];\n if (arg === \"claude\" || arg === \"codex\") {\n result.push(...args.slice(i));\n break;\n }\n if (arg === \"--profile\") {\n i++;\n continue;\n }\n if (arg.startsWith(\"--profile=\")) {\n continue;\n }\n result.push(arg);\n }\n return result;\n}\n\nexport function extractAgentInvocation(args: string[]): { provider: ProviderId; args: string[] } {\n const [agent, ...providerArgs] = args;\n if (agent !== \"claude\" && agent !== \"codex\") {\n throw new Error(\n 'Missing Agent CLI. Use \"dev-anywhere claude ...\" or \"dev-anywhere codex ...\".',\n );\n }\n return { provider: agent, args: providerArgs };\n}\n"],"mappings":"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","names":["relay"]}
package/dist/serve.js CHANGED
@@ -11,10 +11,10 @@ import {
11
11
  decidePtySemanticTransition,
12
12
  extractOscSequences,
13
13
  extractOscSignals
14
- } from "./chunk-OBYEKZWC.js";
14
+ } from "./chunk-Q7AQFYHG.js";
15
15
  import {
16
16
  spawnScript
17
- } from "./chunk-ZUWAB67J.js";
17
+ } from "./chunk-IHUQNJF6.js";
18
18
  import {
19
19
  CLAUDE_PROVIDER,
20
20
  CODEX_PROVIDER,
@@ -35,7 +35,7 @@ import {
35
35
  serializeControl,
36
36
  serializeIpc,
37
37
  serializeWorkerMsg
38
- } from "./chunk-NZZXBVO2.js";
38
+ } from "./chunk-L6J5QCFH.js";
39
39
  import {
40
40
  buildProviderEnv,
41
41
  loadConfig,
@@ -1598,7 +1598,7 @@ var WorkerRegistry = class {
1598
1598
  }
1599
1599
  args.push("--");
1600
1600
  const providerEnv = this.deps.getProviderEnv();
1601
- const child = spawnScript(new URL("../session-worker", import.meta.url), args, {
1601
+ const child = spawnScript("session-worker", args, {
1602
1602
  logger: serviceLogger,
1603
1603
  env: options?.hook ? { ...providerEnv, DEV_ANYWHERE_HOOK_TOKEN: options.hook.token } : providerEnv
1604
1604
  });
@@ -2550,10 +2550,7 @@ var RelayInputHandlers = class {
2550
2550
  })
2551
2551
  );
2552
2552
  if (result.success) {
2553
- serviceLogger.info(
2554
- { sessionId, path, size: result.size },
2555
- "File download handled"
2556
- );
2553
+ serviceLogger.info({ sessionId, path, size: result.size }, "File download handled");
2557
2554
  } else {
2558
2555
  serviceLogger.warn(
2559
2556
  { sessionId, path, errorCode: result.errorCode, error: result.error },
@@ -3584,9 +3581,7 @@ var RelayRouter = class {
3584
3581
  if (sid) {
3585
3582
  const status = this.deps.agentStatusRegistry.get(sid);
3586
3583
  const statuses2 = status && this.deps.sessionManager.getSession(sid) ? [{ sessionId: sid, payload: status }] : [];
3587
- this.deps.relaySend(
3588
- serializeControl({ type: "agent_status_response", requestId, statuses: statuses2 })
3589
- );
3584
+ this.deps.relaySend(serializeControl({ type: "agent_status_response", requestId, statuses: statuses2 }));
3590
3585
  serviceLogger.info({ sessionId: sid, count: statuses2.length }, "Agent status snapshot sent");
3591
3586
  return;
3592
3587
  }
@@ -3595,9 +3590,7 @@ var RelayRouter = class {
3595
3590
  if (!this.deps.sessionManager.getSession(sessionId)) continue;
3596
3591
  statuses.push({ sessionId, payload: status });
3597
3592
  }
3598
- this.deps.relaySend(
3599
- serializeControl({ type: "agent_status_response", requestId, statuses })
3600
- );
3593
+ this.deps.relaySend(serializeControl({ type: "agent_status_response", requestId, statuses }));
3601
3594
  serviceLogger.info({ count: statuses.length }, "Agent status snapshot sent");
3602
3595
  }
3603
3596
  onSessionTerminate(msg) {
@@ -5089,7 +5082,10 @@ async function startService(options) {
5089
5082
  relayConnection.on("connected", () => {
5090
5083
  void controlHandlers.reinitializeOnReconnect().catch((err) => {
5091
5084
  serviceLogger.error(
5092
- { error: err instanceof Error ? err.message : String(err), stack: err instanceof Error ? err.stack : void 0 },
5085
+ {
5086
+ error: err instanceof Error ? err.message : String(err),
5087
+ stack: err instanceof Error ? err.stack : void 0
5088
+ },
5093
5089
  "reinitializeOnReconnect failed: client may see stale state until next manual sync"
5094
5090
  );
5095
5091
  });