@dev-anywhere/proxy 0.1.9 → 0.2.0

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.
Files changed (39) hide show
  1. package/dist/{chunk-BMVYMCKF.js → chunk-3ZUZ22V6.js} +2 -2
  2. package/dist/{chunk-7XMJMVIL.js → chunk-4YQ2JUM7.js} +41 -6
  3. package/dist/chunk-4YQ2JUM7.js.map +1 -0
  4. package/dist/chunk-7UOPAMX7.js +220 -0
  5. package/dist/chunk-7UOPAMX7.js.map +1 -0
  6. package/dist/chunk-NBRBO5GS.js +1032 -0
  7. package/dist/chunk-NBRBO5GS.js.map +1 -0
  8. package/dist/chunk-NQDJ6QAM.js +18 -0
  9. package/dist/chunk-NQDJ6QAM.js.map +1 -0
  10. package/dist/chunk-OBYEKZWC.js +104 -0
  11. package/dist/chunk-OBYEKZWC.js.map +1 -0
  12. package/dist/chunk-PWG6K5QB.js +204 -0
  13. package/dist/chunk-PWG6K5QB.js.map +1 -0
  14. package/dist/{chunk-DCDXAM76.js → chunk-RIQ6OL7X.js} +9 -6
  15. package/dist/chunk-RIQ6OL7X.js.map +1 -0
  16. package/dist/{chunk-6O6JTF24.js → chunk-WUBRUO3G.js} +1 -1
  17. package/dist/index.js +5 -5
  18. package/dist/{relay-token-Z4JZFPQ5.js → relay-token-RKAVVQHE.js} +5 -4
  19. package/dist/{relay-token-Z4JZFPQ5.js.map → relay-token-RKAVVQHE.js.map} +1 -1
  20. package/dist/serve.js +538 -431
  21. package/dist/serve.js.map +1 -1
  22. package/dist/session-worker.js +99 -32
  23. package/dist/session-worker.js.map +1 -1
  24. package/dist/{terminal-FJAIRC73.js → terminal-YO2D2OJU.js} +194 -151
  25. package/dist/terminal-YO2D2OJU.js.map +1 -0
  26. package/package.json +3 -3
  27. package/dist/chunk-2JUB4LDU.js +0 -84
  28. package/dist/chunk-2JUB4LDU.js.map +0 -1
  29. package/dist/chunk-7XMJMVIL.js.map +0 -1
  30. package/dist/chunk-DCDXAM76.js.map +0 -1
  31. package/dist/chunk-ORZTFYXR.js +0 -123
  32. package/dist/chunk-ORZTFYXR.js.map +0 -1
  33. package/dist/chunk-QFYI6AMN.js +0 -870
  34. package/dist/chunk-QFYI6AMN.js.map +0 -1
  35. package/dist/chunk-U5T7ZYXT.js +0 -346
  36. package/dist/chunk-U5T7ZYXT.js.map +0 -1
  37. package/dist/terminal-FJAIRC73.js.map +0 -1
  38. /package/dist/{chunk-BMVYMCKF.js.map → chunk-3ZUZ22V6.js.map} +0 -0
  39. /package/dist/{chunk-6O6JTF24.js.map → chunk-WUBRUO3G.js.map} +0 -0
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/terminal.ts","../src/terminal/tty.ts","../src/terminal/pty-manager.ts","../src/terminal/cwd.ts","../src/terminal/state.ts"],"sourcesContent":["import { connect, type Socket } from \"node:net\";\nimport { existsSync, unlinkSync } from \"node:fs\";\nimport { setTimeout as sleep } from \"node:timers/promises\";\nimport { readTtySize, notifyUser } from \"./terminal/tty.js\";\nimport { PtyManager } from \"./terminal/pty-manager.js\";\nimport { resolveTerminalCwd } from \"./terminal/cwd.js\";\nimport pkg from \"@xterm/headless\";\nconst { Terminal: HeadlessTerminal } = pkg;\nimport { SerializeAddon } from \"@xterm/addon-serialize\";\nimport { UnicodeGraphemesAddon } from \"@xterm/addon-unicode-graphemes\";\nimport {\n extractOscSequences,\n extractOscSignals,\n type PtySemanticState,\n} from \"./common/osc-extractor.js\";\nimport {\n shouldReleaseApprovalWait,\n stateAfterApprovalRelease,\n} from \"./common/pty-approval-state.js\";\nimport { TerminalState, TERMINAL_TRANSITIONS, createExitHandler } from \"./terminal/state.js\";\nimport {\n SOCK_PATH,\n STOPPED_PATH,\n SERVICE_LOG_PATH,\n PROFILE_NAME,\n tildify,\n} from \"./common/paths.js\";\nimport { spawnScript } from \"./common/env.js\";\nimport { daemonRelayArgs } from \"./common/daemon-env.js\";\nimport {\n createIpcReader,\n serializeIpc,\n encodeBinaryIpcFrame,\n type IpcMessage,\n} from \"./ipc/ipc-protocol.js\";\nimport { terminalLogger as log } from \"./common/logger.js\";\nimport { createFSM } from \"./common/state-machine.js\";\nimport {\n CLAUDE_PROVIDER,\n CODEX_PROVIDER,\n type ProviderAdapter,\n type ProviderHookContext,\n type ProviderId,\n} from \"./providers/index.js\";\n\n// serve daemon 自动拉起的连接重试参数\nconst ENSURE_SERVICE_MAX_RETRIES = 20;\nconst ENSURE_SERVICE_INITIAL_DELAY_MS = 100;\nconst ENSURE_SERVICE_MAX_DELAY_MS = 2_000;\n\n// 等待特定类型 IPC 消息的默认超时\nconst WAIT_FOR_MESSAGE_TIMEOUT_MS = 10_000;\n\n// idle 检测:超过 IDLE_THRESHOLD_MS 无输出则翻转 working -> turn_complete\nconst IDLE_CHECK_INTERVAL_MS = 3_000;\nconst IDLE_THRESHOLD_MS = 3_000;\n\n// serve 连接断开后的重连重试参数\nconst RECONNECT_INITIAL_DELAY_MS = 1_000;\nconst RECONNECT_MAX_DELAY_MS = 5_000;\n// 连续 spawn 失败到达阈值后停止自动 spawn,降为被动 tryConnect 轮询。\n// 作用:环境异常(端口占用、依赖缺失、权限不足)时避免反复拉起短命子进程把日志刷爆。\nconst SPAWN_FAILURE_THRESHOLD = 3;\n\nconst PROVIDERS: Record<ProviderId, ProviderAdapter> = {\n claude: CLAUDE_PROVIDER,\n codex: CODEX_PROVIDER,\n};\n\nfunction tryConnect(sockPath: string): Promise<Socket | null> {\n return new Promise((resolve) => {\n const s = connect(sockPath);\n s.on(\"connect\", () => resolve(s));\n s.on(\"error\", () => resolve(null));\n });\n}\n\nasync function ensureService(autoStart = true): Promise<Socket> {\n const existing = await tryConnect(SOCK_PATH);\n if (existing) {\n log.info(\"Connected to existing service\");\n return existing;\n }\n\n if (!autoStart) throw new Error(\"Service is not running\");\n\n if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);\n\n log.info(\"Auto-starting serve daemon\");\n const child = spawnScript(\n new URL(\"./serve\", import.meta.url),\n [\"--profile\", PROFILE_NAME, ...daemonRelayArgs()],\n {\n env: { ...process.env },\n logger: log,\n },\n );\n\n // 监听 daemon 失败信号,让下面的 tryConnect 轮询能在 daemon 启动时就崩的场景下立刻抛诊断。\n // - 'exit':进程启动成功后又退出(配置错误、端口占用、内部崩溃),带 code/signal。\n // - 'error':spawn 本身失败(ENOENT 找不到 tsx/node 等),Node 文档说此时 'exit' may or may not 跟着 fire,\n // 所以显式监听补完备性。spawnScript 内部另装了一对只管日志的监听器,跟这里互不影响。\n let childFailed = false;\n let exitCode: number | null = null;\n let exitSignal: NodeJS.Signals | null = null;\n let spawnError: Error | null = null;\n child.once(\"exit\", (code, signal) => {\n childFailed = true;\n exitCode = code;\n exitSignal = signal;\n });\n child.once(\"error\", (err) => {\n childFailed = true;\n spawnError = err;\n });\n\n for (let i = 0; i < ENSURE_SERVICE_MAX_RETRIES; i++) {\n const delay = Math.min(ENSURE_SERVICE_INITIAL_DELAY_MS * (i + 1), ENSURE_SERVICE_MAX_DELAY_MS);\n await sleep(delay);\n\n if (childFailed) {\n log.error(\n { code: exitCode, signal: exitSignal, err: spawnError && String(spawnError) },\n \"Serve daemon failed to start\",\n );\n const detail = spawnError\n ? `spawn error=${String(spawnError)}`\n : `code=${exitCode}, signal=${exitSignal}`;\n throw new Error(\n `Serve daemon failed to start (${detail}). Check ${SERVICE_LOG_PATH} for details.`,\n );\n }\n\n const socket = await tryConnect(SOCK_PATH);\n if (socket) {\n log.info({ attempt: i + 1 }, \"Connected to service after retry\");\n return socket;\n }\n }\n\n log.error({ maxRetries: ENSURE_SERVICE_MAX_RETRIES }, \"Failed to connect to service\");\n throw new Error(\n `Failed to connect to dev-anywhere service after ${ENSURE_SERVICE_MAX_RETRIES} retries. Check ${SERVICE_LOG_PATH} for details.`,\n );\n}\n\nfunction waitForMessage<T extends IpcMessage[\"type\"]>(\n socket: Socket,\n messageType: T,\n): Promise<Extract<IpcMessage, { type: T }>> {\n return new Promise((resolve, reject) => {\n let timeout: NodeJS.Timeout | null = null;\n const dispose = createIpcReader(socket, (msg: IpcMessage) => {\n if (msg.type === messageType) {\n if (timeout) clearTimeout(timeout);\n dispose();\n resolve(msg as Extract<IpcMessage, { type: T }>);\n }\n });\n timeout = setTimeout(() => {\n dispose();\n reject(new Error(`Timeout waiting for ${messageType}`));\n }, WAIT_FOR_MESSAGE_TIMEOUT_MS);\n });\n}\n\nclass TerminalSession {\n private readonly fsm = createFSM<TerminalState>({\n initial: TerminalState.INIT,\n transitions: TERMINAL_TRANSITIONS,\n onTransition: (from, to) => log.info({ from, to }, \"Terminal state transition\"),\n });\n private readonly sessionCwd = resolveTerminalCwd();\n // socket 在 run() 中连上 serve 后首次赋值;reconnect 会重新赋值为新实例\n private socket!: Socket;\n private sessionId: string | null = null;\n private hookContext: ProviderHookContext | null = null;\n private ptyManager: PtyManager | null = null;\n private lastOutputTime = 0;\n private idleCheckTimer: NodeJS.Timeout | null = null;\n private currentPtyState: PtySemanticState = \"turn_complete\";\n // headless terminal 在本进程维护,用于按需 serialize() 给远程 client\n private headlessTerminal: InstanceType<typeof HeadlessTerminal> | null = null;\n private serializeAddon: SerializeAddon | null = null;\n private outputSeq = 0;\n private remoteDetached = false;\n // 记录上次 bridge 状态避免重连抖动导致 banner 连刷;初值 null 让首次状态(无论真假)都打,启动时提示 remote viewing 是否就绪\n private lastBridgeConnected: boolean | null = null;\n // 收尾函数在 run() 里创建一次,PTY 退出与 SIGTERM 共用;内部通过 fsm EXITED 检查短路\n private cleanupAndExit!: (code: number) => void;\n\n constructor(\n private readonly provider: ProviderAdapter,\n private readonly providerArgs: string[],\n ) {}\n\n async run(): Promise<void> {\n log.info(\"Terminal starting\");\n this.fsm.transitionTo(TerminalState.CONNECTING_SERVICE);\n this.socket = await ensureService();\n\n await this.createSession();\n this.initHeadlessTerminal();\n this.cleanupAndExit = createExitHandler({\n fsm: this.fsm,\n getSocket: () => this.socket,\n getSessionId: () => this.sessionId,\n getIdleCheckTimer: () => this.idleCheckTimer,\n });\n\n this.setupSocketHandlers();\n this.startPtyManager();\n\n this.socket.write(\n serializeIpc({ type: \"pty_register\", sessionId: this.sessionId!, pid: process.pid }),\n );\n this.replayCurrentPtyState();\n this.fsm.transitionTo(TerminalState.RUNNING);\n this.setupIdleCheck();\n\n process.on(\"SIGTERM\", () => {\n log.info({ sessionId: this.sessionId }, \"SIGTERM received, shutting down\");\n this.cleanupAndExit(143);\n });\n }\n\n private async createSession(): Promise<void> {\n this.fsm.transitionTo(TerminalState.CREATING_SESSION);\n const responsePromise = waitForMessage(this.socket, \"session_create_response\");\n this.socket.write(\n serializeIpc({\n type: \"session_create_request\",\n mode: \"pty\",\n provider: this.provider.id,\n cwd: this.sessionCwd,\n name: tildify(this.sessionCwd),\n pid: process.pid,\n }),\n );\n const response = await responsePromise;\n if (response.error) {\n throw new Error(`Failed to create session: ${response.error}`);\n }\n this.sessionId = response.sessionId;\n this.hookContext = response.hook ?? null;\n }\n\n private initHeadlessTerminal(): void {\n const { cols, rows } = readTtySize(process.stdout);\n log.info(\n { sessionId: this.sessionId, cols, rows },\n \"Session created, initializing headless terminal\",\n );\n this.headlessTerminal = new HeadlessTerminal({\n cols,\n rows,\n scrollback: 5000,\n allowProposedApi: true,\n });\n this.serializeAddon = new SerializeAddon();\n // UnicodeGraphemesAddon activate() 里会设置 activeVersion = '15-graphemes'\n this.headlessTerminal.loadAddon(this.serializeAddon);\n this.headlessTerminal.loadAddon(new UnicodeGraphemesAddon());\n }\n\n private startPtyManager(): void {\n this.ptyManager = new PtyManager({\n provider: this.provider,\n providerArgs: this.providerArgs,\n cwd: this.sessionCwd,\n hook: this.hookContext ?? undefined,\n tap: (data) => this.handlePtyData(data),\n stdin: process.stdin,\n stdout: process.stdout,\n onResize: (newCols, newRows) => {\n if (this.headlessTerminal) this.headlessTerminal.resize(newCols, newRows);\n if (this.socket.writable && this.sessionId) {\n this.socket.write(\n serializeIpc({\n type: \"pty_resize\",\n sessionId: this.sessionId,\n cols: newCols,\n rows: newRows,\n }),\n );\n }\n },\n onSessionExit: (code: number) => {\n log.info({ sessionId: this.sessionId, exitCode: code }, \"PTY exited, cleaning up\");\n this.cleanupAndExit(code);\n },\n });\n this.ptyManager.start();\n log.info({ sessionId: this.sessionId }, \"PTY started with headless terminal\");\n }\n\n // PTY 的每一帧输出都要:追到 headless terminal 状态、推 binary IPC、提取 provider 语义事件\n private handlePtyData(data: string): void {\n this.lastOutputTime = Date.now();\n this.outputSeq += 1;\n\n if (this.headlessTerminal) this.headlessTerminal.write(data);\n\n if (!this.remoteDetached && this.socket.writable && this.sessionId) {\n this.socket.write(\n encodeBinaryIpcFrame(this.sessionId, Buffer.from(data, \"utf-8\"), this.outputSeq),\n );\n }\n\n const oscSequences = extractOscSequences(data);\n const signal = extractOscSignals(data, this.provider.id);\n if (oscSequences.length > 0) {\n log.debug(\n {\n sessionId: this.sessionId,\n oscSequences,\n signal,\n },\n \"PTY OSC sequences parsed\",\n );\n }\n if (signal?.title) {\n this.sendTerminalTitle(signal.title);\n }\n if (signal?.state === \"approval_wait\") {\n this.currentPtyState = \"approval_wait\";\n this.sendPtyState(\"approval_wait\", { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (\n shouldReleaseApprovalWait({\n currentState: this.currentPtyState,\n signalState: signal?.state,\n })\n ) {\n const nextState = stateAfterApprovalRelease(signal?.state);\n this.currentPtyState = nextState;\n this.sendPtyState(nextState, { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (this.currentPtyState === \"approval_wait\" && signal?.state !== \"turn_complete\") {\n this.sendPtyState(\"approval_wait\", { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (signal && signal.state !== \"working\") {\n this.currentPtyState = signal.state;\n this.sendPtyState(signal.state, { title: signal.title, tool: signal.tool });\n return;\n }\n if (this.currentPtyState !== \"working\") {\n this.currentPtyState = \"working\";\n this.sendPtyState(\"working\");\n }\n }\n\n private sendTerminalTitle(title: string): void {\n if (this.remoteDetached || !this.socket.writable || !this.sessionId) return;\n this.socket.write(\n serializeIpc({\n type: \"pty_title_change\",\n sessionId: this.sessionId,\n title,\n }),\n );\n }\n\n private sendPtyState(state: PtySemanticState, meta?: { title?: string; tool?: string }): void {\n if (this.remoteDetached || !this.socket.writable || !this.sessionId) return;\n this.socket.write(\n serializeIpc({\n type: \"pty_semantic_event\",\n sessionId: this.sessionId,\n state,\n ...(meta?.title !== undefined ? { title: meta.title } : {}),\n ...(meta?.tool !== undefined ? { tool: meta.tool } : {}),\n }),\n );\n log.info(\n { sessionId: this.sessionId, state, title: meta?.title, tool: meta?.tool },\n \"PTY semantic event pushed\",\n );\n }\n\n private replayCurrentPtyState(): void {\n if (this.currentPtyState === \"turn_complete\") return;\n this.sendPtyState(this.currentPtyState);\n }\n\n private handleBridgeStatus(connected: boolean): void {\n if (this.remoteDetached) return;\n if (this.lastBridgeConnected === connected) return;\n this.lastBridgeConnected = connected;\n log.info({ connected }, \"Bridge status changed, notifying user\");\n notifyUser(connected ? \"relay online\" : \"relay offline — remote viewing unavailable\");\n }\n\n private setupSocketHandlers(): void {\n createIpcReader(this.socket, (msg: IpcMessage) => {\n if (msg.type === \"pty_input\" && msg.sessionId === this.sessionId) {\n log.debug({ sessionId: this.sessionId, bytes: msg.data.length }, \"Remote input received\");\n this.ptyManager?.write(msg.data);\n } else if (msg.type === \"pty_detach\" && msg.sessionId === this.sessionId) {\n this.detachRemoteView();\n } else if (msg.type === \"bridge_status\") {\n this.handleBridgeStatus(msg.connected);\n } else if (msg.type === \"pty_subscribe\" && msg.sessionId === this.sessionId) {\n if (this.serializeAddon && this.headlessTerminal) {\n const data = this.serializeAddon.serialize();\n this.socket.write(\n serializeIpc({\n type: \"pty_snapshot\",\n sessionId: msg.sessionId,\n cols: this.headlessTerminal.cols,\n rows: this.headlessTerminal.rows,\n data,\n outputSeq: this.outputSeq,\n requestId: msg.requestId,\n }),\n );\n log.info(\n {\n sessionId: this.sessionId,\n cols: this.headlessTerminal.cols,\n rows: this.headlessTerminal.rows,\n bytes: data.length,\n },\n \"Snapshot sent via IPC\",\n );\n }\n }\n });\n\n this.socket.on(\"close\", () => {\n log.info(\"Serve socket closed\");\n if (this.remoteDetached) {\n log.info(\"Remote view detached, skipping serve reconnect\");\n return;\n }\n if (!this.fsm.isIn([TerminalState.RECONNECTING, TerminalState.EXITED])) {\n this.fsm.transitionTo(TerminalState.RECONNECTING);\n this.reconnectToServe();\n }\n });\n\n // socket error 通常和 close 成对出现;这里只记 warn 避免静默吞错,重连仍由 close handler 触发\n this.socket.on(\"error\", (err) => {\n log.warn({ err: err.message }, \"Serve socket error\");\n });\n }\n\n // 超过 IDLE_THRESHOLD_MS 无 PTY 输出则从 working 翻回 turn_complete\n private setupIdleCheck(): void {\n if (this.idleCheckTimer) clearInterval(this.idleCheckTimer);\n this.idleCheckTimer = setInterval(() => {\n if (this.lastOutputTime > 0 && Date.now() - this.lastOutputTime > IDLE_THRESHOLD_MS) {\n this.lastOutputTime = 0;\n if (this.currentPtyState === \"working\") {\n this.currentPtyState = \"turn_complete\";\n this.sendPtyState(\"turn_complete\");\n }\n }\n }, IDLE_CHECK_INTERVAL_MS);\n }\n\n private async reconnectToServe(): Promise<void> {\n log.info(\"Serve connection lost, starting reconnection\");\n\n // 两条路径都不该再继续 spawn daemon:\n // - STOPPED=true:用户主动 dev-anywhere stop,不要对抗用户意图。\n // - consecutiveSpawnFailures 跨过阈值:说明环境有持续性问题,spawn 再多也白搭。\n // 进入 passive 后仅做 tryConnect 等待,daemon 起来或用户 dev-anywhere start 后自动恢复。\n let consecutiveSpawnFailures = 0;\n\n for (let i = 0; ; i++) {\n if (this.remoteDetached) return;\n await sleep(Math.min(RECONNECT_INITIAL_DELAY_MS * (i + 1), RECONNECT_MAX_DELAY_MS));\n\n const stopped = existsSync(STOPPED_PATH);\n const degraded = consecutiveSpawnFailures >= SPAWN_FAILURE_THRESHOLD;\n const passive = stopped || degraded;\n\n try {\n log.debug({ attempt: i + 1, stopped, degraded }, \"Reconnect attempt\");\n const newSocket = passive ? await tryConnect(SOCK_PATH) : await ensureService();\n if (!newSocket) continue;\n\n if (degraded) notifyUser(\"serve daemon reachable, reconnected\");\n consecutiveSpawnFailures = 0;\n\n this.socket = newSocket;\n log.info({ attempt: i + 1, sessionId: this.sessionId }, \"Reconnected to serve\");\n\n this.setupSocketHandlers();\n\n if (this.sessionId) {\n this.fsm.transitionTo(TerminalState.CREATING_SESSION);\n this.socket.write(\n serializeIpc({\n type: \"session_create_request\",\n mode: \"pty\",\n provider: this.provider.id,\n cwd: this.sessionCwd,\n name: tildify(this.sessionCwd),\n pid: process.pid,\n sessionId: this.sessionId,\n }),\n );\n const resp = await waitForMessage(this.socket, \"session_create_response\");\n if (!resp.error) {\n this.sessionId = resp.sessionId;\n this.socket.write(\n serializeIpc({ type: \"pty_register\", sessionId: this.sessionId, pid: process.pid }),\n );\n this.replayCurrentPtyState();\n this.fsm.transitionTo(TerminalState.RUNNING);\n log.info({ sessionId: this.sessionId }, \"Session re-registered after reconnect\");\n }\n } else {\n this.fsm.transitionTo(TerminalState.RUNNING);\n }\n\n return;\n } catch (err) {\n // passive 模式走 tryConnect,失败返回 null 不抛;这里只可能是 ensureService spawn 失败\n if (!passive) {\n consecutiveSpawnFailures++;\n if (consecutiveSpawnFailures === SPAWN_FAILURE_THRESHOLD) {\n notifyUser(\n `serve daemon spawn failed ${SPAWN_FAILURE_THRESHOLD}x — auto-spawn disabled; check environment or run 'dev-anywhere start'`,\n );\n }\n }\n log.debug(\n { err: err instanceof Error ? err.message : err, attempt: i + 1, degraded },\n \"Reconnect attempt failed\",\n );\n }\n }\n }\n\n private detachRemoteView(): void {\n const sessionId = this.sessionId;\n if (!sessionId) return;\n this.remoteDetached = true;\n this.sessionId = null;\n this.hookContext = null;\n this.currentPtyState = \"turn_complete\";\n log.info({ sessionId }, \"Remote view detached; local PTY keeps running\");\n notifyUser(\"remote viewing detached\");\n if (this.socket.writable) this.socket.end();\n }\n}\n\nfunction providerFromEnv(): ProviderId {\n return process.env.DEV_ANYWHERE_PROVIDER === \"codex\" ? \"codex\" : \"claude\";\n}\n\nexport async function startTerminal(\n providerArgs: string[],\n providerId: ProviderId = providerFromEnv(),\n): Promise<void> {\n await new TerminalSession(PROVIDERS[providerId], providerArgs).run();\n}\n","// 读 stdout cols/rows,非 TTY 抛错。\nexport function readTtySize(stream: NodeJS.WriteStream): { cols: number; rows: number } {\n const { columns, rows } = stream;\n if (columns === undefined || rows === undefined) {\n throw new Error(\n \"stdout is not an interactive TTY (columns/rows undefined); dev-anywhere requires running in a real terminal\",\n );\n }\n return { cols: columns, rows };\n}\n\n// 发一条 OSC 9 iTerm2-style 系统通知 + 响铃。iTerm2 / kitty / wezterm 等会弹出带 message\n// 的系统通知;不认 OSC 9 的终端会忽略转义序列只剩下 BEL 响铃。\n// 用此而非 stderr banner 的原因:dev-anywhere 对 Claude PTY 画面保持透明是硬约束,\n// banner 会挤掉 Claude 的渲染行,OSC 9 不占画面,BEL 是纯听觉信号。\nexport function notifyUser(message: string): void {\n process.stderr.write(`\\x1b]9;${message}\\x07`);\n}\n\n// Provider TUI 可能开启 bracketed paste、application cursor/keypad、mouse tracking、\n// xterm modifyOtherKeys 或 kitty keyboard protocol。若 provider 被远程终止或异常退出,\n// 这些模式可能来不及自行恢复,外层 shell 会把 Ctrl-C 显示成 \";5;99~\" 一类残留序列。\nexport function restoreHostTerminalModes(stream: NodeJS.WriteStream): void {\n if (!stream.isTTY) return;\n const restoreSequences = [\n \"\\x1b[?1l\", // application cursor keys off\n \"\\x1b>\", // application keypad off\n \"\\x1b[?1000l\",\n \"\\x1b[?1002l\",\n \"\\x1b[?1003l\",\n \"\\x1b[?1004l\",\n \"\\x1b[?1006l\",\n \"\\x1b[?1015l\",\n \"\\x1b[?2004l\", // bracketed paste off\n \"\\x1b[>4;0m\", // xterm modifyOtherKeys off\n \"\\x1b[<u\", // kitty keyboard protocol off\n ].join(\"\");\n stream.write(restoreSequences);\n}\n","import * as pty from \"node-pty\";\nimport type { IPty } from \"node-pty\";\nimport type { ProviderAdapter, ProviderHookContext } from \"../providers/index.js\";\nimport { readTtySize, restoreHostTerminalModes } from \"./tty.js\";\n\ninterface PtyManagerOptions {\n provider: ProviderAdapter;\n providerArgs: string[];\n cwd: string;\n hook?: ProviderHookContext;\n tap: (data: string) => void;\n stdin: NodeJS.ReadStream;\n stdout: NodeJS.WriteStream;\n onSessionExit?: (code: number) => void | Promise<void>;\n onResize?: (cols: number, rows: number) => void;\n}\n\nexport class PtyManager {\n private child: IPty | null = null;\n private readonly provider: ProviderAdapter;\n private readonly providerArgs: string[];\n private readonly cwd: string;\n private readonly hook?: ProviderHookContext;\n private readonly tap: (data: string) => void;\n private readonly stdin: NodeJS.ReadStream;\n private readonly stdout: NodeJS.WriteStream;\n private readonly onSessionExit?: (code: number) => void;\n private readonly onResize?: (cols: number, rows: number) => void;\n\n constructor(options: PtyManagerOptions) {\n this.provider = options.provider;\n this.providerArgs = options.providerArgs;\n this.cwd = options.cwd;\n this.hook = options.hook;\n this.tap = options.tap;\n this.stdin = options.stdin;\n this.stdout = options.stdout;\n this.onSessionExit = options.onSessionExit;\n this.onResize = options.onResize;\n }\n\n start(): void {\n const { cols, rows } = readTtySize(this.stdout);\n\n const command = this.provider.buildTerminalCommand(\n { args: this.providerArgs, hook: this.hook },\n process.env,\n );\n const child = pty.spawn(command.command, command.args, {\n name: process.env.TERM ?? \"xterm-256color\",\n cols,\n rows,\n cwd: this.cwd,\n env: command.env as Record<string, string>,\n });\n this.child = child;\n\n // raw mode 仅在 stdin 为 TTY 时开启\n const isInteractive = this.stdin.isTTY === true;\n if (isInteractive) {\n this.stdin.setRawMode(true);\n }\n this.stdin.resume();\n\n // stdin -> PTY\n this.stdin.on(\"data\", (data: Buffer) => {\n child.write(data.toString());\n });\n\n // PTY -> stdout + tap\n child.onData((data: string) => this.handleData(data));\n\n // resize 防抖,50ms 窗口合并快速连续的尺寸变化\n let resizeTimer: ReturnType<typeof setTimeout> | null = null;\n this.stdout.on(\"resize\", () => {\n if (resizeTimer) clearTimeout(resizeTimer);\n resizeTimer = setTimeout(() => {\n const { cols: newCols, rows: newRows } = readTtySize(this.stdout);\n child.resize(newCols, newRows);\n this.onResize?.(newCols, newRows);\n }, 50);\n });\n\n // 子进程退出,按 Unix 惯例处理信号退出码,通过回调通知调用方\n child.onExit(({ exitCode, signal }) => {\n if (isInteractive) {\n try {\n this.stdin.setRawMode(false);\n } catch {\n // stdin 可能已关闭\n }\n }\n restoreHostTerminalModes(this.stdout);\n const code = signal ? 128 + signal : exitCode;\n this.onSessionExit?.(code);\n });\n\n // stdin 结束时写入 EOF 控制字符到 PTY\n this.stdin.on(\"end\", () => {\n child.write(\"\\x04\");\n });\n }\n\n /**\n * PTY 数据到达时的统一处理:OSC 9 修复 + 输出到终端 + 传给 tap\n */\n private handleData(data: string): void {\n // PTY 的 onlcr 会把 OSC 序列里的 \\n 转成 \\r\\n,还原为 \\n\n const fixed = data.replace(\n // eslint-disable-next-line no-control-regex\n /\\x1b\\]9;([\\s\\S]*?)\\x07/g,\n (_, content: string) => `\\x1b]9;${content.replace(/\\r\\n/g, \"\\n\")}\\x07`,\n );\n this.stdout.write(fixed);\n this.tap(data);\n }\n\n // 向 PTY 子进程写入数据,用于远程输入注入\n write(data: string): void {\n this.child?.write(data);\n }\n\n cleanup(exitCode: number): void {\n if (this.stdin.isTTY) {\n try {\n this.stdin.setRawMode(false);\n } catch {\n // stdin 可能已关闭\n }\n }\n restoreHostTerminalModes(this.stdout);\n if (this.child) {\n try {\n this.child.kill();\n } catch {\n // 子进程可能已退出\n }\n }\n this.onSessionExit?.(exitCode);\n }\n}\n","import { statSync } from \"node:fs\";\n\nfunction isDirectory(path: string | undefined): path is string {\n if (!path) return false;\n try {\n return statSync(path).isDirectory();\n } catch {\n return false;\n }\n}\n\nexport function resolveTerminalCwd(env: NodeJS.ProcessEnv = process.env): string {\n const candidates = [env.DEV_ANYWHERE_CWD, env.INIT_CWD, env.PWD, process.cwd()];\n return candidates.find(isDirectory) ?? process.cwd();\n}\n","import type { Socket } from \"node:net\";\nimport { createFSM } from \"../common/state-machine.js\";\nimport { serializeIpc } from \"../ipc/ipc-protocol.js\";\n\n// terminal 进程生命周期状态\nexport const TerminalState = {\n INIT: \"init\",\n CONNECTING_SERVICE: \"connecting_service\",\n CREATING_SESSION: \"creating_session\",\n RUNNING: \"running\",\n RECONNECTING: \"reconnecting\",\n EXITED: \"exited\",\n} as const;\nexport type TerminalState = (typeof TerminalState)[keyof typeof TerminalState];\n\n// 允许的状态转换。CREATING_SESSION/RUNNING 下可被 socket close 打断进入 RECONNECTING;\n// 任意非终态都可能被 PTY 退出或 SIGTERM 打断进入 EXITED。\nexport const TERMINAL_TRANSITIONS: Record<TerminalState, readonly TerminalState[]> = {\n init: [\"connecting_service\"],\n connecting_service: [\"creating_session\", \"exited\"],\n creating_session: [\"running\", \"reconnecting\", \"exited\"],\n running: [\"reconnecting\", \"exited\"],\n reconnecting: [\"creating_session\", \"running\", \"exited\"],\n exited: [],\n};\n\n// 下面三个依赖是 getter 而非值:因为它们在 terminal.ts 里是 let 变量,在 handler 创建之后还会变——\n// socket 在 reconnect 时被重新赋值为新实例,sessionId 在 session_create 成功后才有值,\n// idleCheckTimer 在 setupIdleCheck 跑完才赋值。直接传值只会记录 handler 构造那一刻的旧值。\ninterface ExitHandlerDeps {\n fsm: ReturnType<typeof createFSM<TerminalState>>;\n getSocket: () => Socket;\n getSessionId: () => string | null;\n getIdleCheckTimer: () => NodeJS.Timeout | null;\n // 测试注入点,production 默认 process.exit\n exit?: (code: number) => void;\n}\n\n// 构造统一的收尾函数:转 EXITED → 停 idle 定时器 → 给 serve 发 pty_deregister → 退进程。\n// onSessionExit 与 SIGTERM handler 共享同一实例;Ctrl+C 两连击或 PTY 退出与 SIGTERM 竞争时,\n// 第二次调用通过 fsm EXITED 检查直接短路。\nexport function createExitHandler(deps: ExitHandlerDeps): (code: number) => void {\n const exit = deps.exit ?? ((code: number) => process.exit(code));\n return (code: number) => {\n if (deps.fsm.is(TerminalState.EXITED)) return;\n deps.fsm.transitionTo(TerminalState.EXITED);\n const timer = deps.getIdleCheckTimer();\n if (timer) clearInterval(timer);\n const socket = deps.getSocket();\n const sessionId = deps.getSessionId();\n if (socket.writable && sessionId) {\n socket.end(serializeIpc({ type: \"pty_deregister\", sessionId }), () => exit(code));\n } else {\n exit(code);\n }\n };\n}\n"],"mappings":"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","names":[]}