mstro-app 0.4.52 ā 0.5.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +129 -190
- package/PRIVACY.md +3 -3
- package/README.md +15 -6
- package/bin/commands/config.js +0 -1
- package/bin/mstro.js +1 -2
- package/bin/postinstall.js +0 -1
- package/dist/server/cli/headless/claude-invoker-process.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker-process.js +0 -1
- package/dist/server/cli/headless/claude-invoker-process.js.map +1 -1
- package/dist/server/cli/headless/claude-invoker-stall.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker-stall.js +7 -3
- package/dist/server/cli/headless/claude-invoker-stall.js.map +1 -1
- package/dist/server/cli/headless/claude-invoker-stream.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker-stream.js +0 -1
- package/dist/server/cli/headless/claude-invoker-stream.js.map +1 -1
- package/dist/server/cli/headless/claude-invoker-tools.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker-tools.js +0 -1
- package/dist/server/cli/headless/claude-invoker-tools.js.map +1 -1
- package/dist/server/cli/headless/claude-invoker.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker.js +1 -2
- package/dist/server/cli/headless/claude-invoker.js.map +1 -1
- package/dist/server/cli/headless/haiku-assessments.d.ts.map +1 -1
- package/dist/server/cli/headless/haiku-assessments.js +0 -1
- package/dist/server/cli/headless/haiku-assessments.js.map +1 -1
- package/dist/server/cli/headless/headless-logger.d.ts.map +1 -1
- package/dist/server/cli/headless/headless-logger.js +0 -1
- package/dist/server/cli/headless/headless-logger.js.map +1 -1
- package/dist/server/cli/headless/index.d.ts.map +1 -1
- package/dist/server/cli/headless/index.js +0 -1
- package/dist/server/cli/headless/index.js.map +1 -1
- package/dist/server/cli/headless/native-timeout-detector.d.ts.map +1 -1
- package/dist/server/cli/headless/native-timeout-detector.js +0 -1
- package/dist/server/cli/headless/native-timeout-detector.js.map +1 -1
- package/dist/server/cli/headless/output-utils.d.ts.map +1 -1
- package/dist/server/cli/headless/output-utils.js +0 -1
- package/dist/server/cli/headless/output-utils.js.map +1 -1
- package/dist/server/cli/headless/prompt-utils.d.ts.map +1 -1
- package/dist/server/cli/headless/prompt-utils.js +0 -1
- package/dist/server/cli/headless/prompt-utils.js.map +1 -1
- package/dist/server/cli/headless/resilient-runner.d.ts.map +1 -1
- package/dist/server/cli/headless/resilient-runner.js +0 -1
- package/dist/server/cli/headless/resilient-runner.js.map +1 -1
- package/dist/server/cli/headless/retry-strategies.d.ts.map +1 -1
- package/dist/server/cli/headless/retry-strategies.js +0 -1
- package/dist/server/cli/headless/retry-strategies.js.map +1 -1
- package/dist/server/cli/headless/runner.d.ts.map +1 -1
- package/dist/server/cli/headless/runner.js +63 -68
- package/dist/server/cli/headless/runner.js.map +1 -1
- package/dist/server/cli/headless/stall-assessor.d.ts.map +1 -1
- package/dist/server/cli/headless/stall-assessor.js +9 -5
- package/dist/server/cli/headless/stall-assessor.js.map +1 -1
- package/dist/server/cli/headless/tool-watchdog.d.ts.map +1 -1
- package/dist/server/cli/headless/tool-watchdog.js +0 -1
- package/dist/server/cli/headless/tool-watchdog.js.map +1 -1
- package/dist/server/cli/headless/types.d.ts.map +1 -1
- package/dist/server/cli/headless/types.js +0 -1
- package/dist/server/cli/headless/types.js.map +1 -1
- package/dist/server/cli/improvisation-attachments.d.ts.map +1 -1
- package/dist/server/cli/improvisation-attachments.js +0 -1
- package/dist/server/cli/improvisation-attachments.js.map +1 -1
- package/dist/server/cli/improvisation-history-store.d.ts +16 -0
- package/dist/server/cli/improvisation-history-store.d.ts.map +1 -0
- package/dist/server/cli/improvisation-history-store.js +51 -0
- package/dist/server/cli/improvisation-history-store.js.map +1 -0
- package/dist/server/cli/improvisation-movements.d.ts +31 -0
- package/dist/server/cli/improvisation-movements.d.ts.map +1 -0
- package/dist/server/cli/improvisation-movements.js +92 -0
- package/dist/server/cli/improvisation-movements.js.map +1 -0
- package/dist/server/cli/improvisation-output-queue.d.ts +13 -0
- package/dist/server/cli/improvisation-output-queue.d.ts.map +1 -0
- package/dist/server/cli/improvisation-output-queue.js +39 -0
- package/dist/server/cli/improvisation-output-queue.js.map +1 -0
- package/dist/server/cli/improvisation-retry.d.ts +21 -51
- package/dist/server/cli/improvisation-retry.d.ts.map +1 -1
- package/dist/server/cli/improvisation-retry.js +18 -434
- package/dist/server/cli/improvisation-retry.js.map +1 -1
- package/dist/server/cli/improvisation-session-manager.d.ts +10 -8
- package/dist/server/cli/improvisation-session-manager.d.ts.map +1 -1
- package/dist/server/cli/improvisation-session-manager.js +53 -149
- package/dist/server/cli/improvisation-session-manager.js.map +1 -1
- package/dist/server/cli/improvisation-types.d.ts.map +1 -1
- package/dist/server/cli/improvisation-types.js +0 -1
- package/dist/server/cli/improvisation-types.js.map +1 -1
- package/dist/server/cli/retry/retry-best-result.d.ts +4 -0
- package/dist/server/cli/retry/retry-best-result.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-best-result.js +60 -0
- package/dist/server/cli/retry/retry-best-result.js.map +1 -0
- package/dist/server/cli/retry/retry-context-loss.d.ts +6 -0
- package/dist/server/cli/retry/retry-context-loss.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-context-loss.js +67 -0
- package/dist/server/cli/retry/retry-context-loss.js.map +1 -0
- package/dist/server/cli/retry/retry-premature-completion.d.ts +5 -0
- package/dist/server/cli/retry/retry-premature-completion.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-premature-completion.js +80 -0
- package/dist/server/cli/retry/retry-premature-completion.js.map +1 -0
- package/dist/server/cli/retry/retry-recovery-strategies.d.ts +13 -0
- package/dist/server/cli/retry/retry-recovery-strategies.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-recovery-strategies.js +165 -0
- package/dist/server/cli/retry/retry-recovery-strategies.js.map +1 -0
- package/dist/server/cli/retry/retry-resume-strategy.d.ts +12 -0
- package/dist/server/cli/retry/retry-resume-strategy.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-resume-strategy.js +21 -0
- package/dist/server/cli/retry/retry-resume-strategy.js.map +1 -0
- package/dist/server/cli/retry/retry-runner-factory.d.ts +11 -0
- package/dist/server/cli/retry/retry-runner-factory.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-runner-factory.js +59 -0
- package/dist/server/cli/retry/retry-runner-factory.js.map +1 -0
- package/dist/server/cli/retry/retry-tool-results.d.ts +9 -0
- package/dist/server/cli/retry/retry-tool-results.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-tool-results.js +23 -0
- package/dist/server/cli/retry/retry-tool-results.js.map +1 -0
- package/dist/server/cli/retry/retry-types.d.ts +30 -0
- package/dist/server/cli/retry/retry-types.d.ts.map +1 -0
- package/dist/server/cli/retry/retry-types.js +3 -0
- package/dist/server/cli/retry/retry-types.js.map +1 -0
- package/dist/server/index.js +21 -110
- package/dist/server/index.js.map +1 -1
- package/dist/server/mcp/bouncer-cli.js +0 -1
- package/dist/server/mcp/bouncer-cli.js.map +1 -1
- package/dist/server/mcp/bouncer-haiku.d.ts.map +1 -1
- package/dist/server/mcp/bouncer-haiku.js +0 -1
- package/dist/server/mcp/bouncer-haiku.js.map +1 -1
- package/dist/server/mcp/bouncer-integration.d.ts.map +1 -1
- package/dist/server/mcp/bouncer-integration.js +0 -1
- package/dist/server/mcp/bouncer-integration.js.map +1 -1
- package/dist/server/mcp/security-analysis.d.ts.map +1 -1
- package/dist/server/mcp/security-analysis.js +0 -1
- package/dist/server/mcp/security-analysis.js.map +1 -1
- package/dist/server/mcp/security-audit.d.ts.map +1 -1
- package/dist/server/mcp/security-audit.js +0 -1
- package/dist/server/mcp/security-audit.js.map +1 -1
- package/dist/server/mcp/security-patterns.d.ts.map +1 -1
- package/dist/server/mcp/security-patterns.js +0 -1
- package/dist/server/mcp/security-patterns.js.map +1 -1
- package/dist/server/mcp/server.js +0 -1
- package/dist/server/mcp/server.js.map +1 -1
- package/dist/server/routes/files.d.ts.map +1 -1
- package/dist/server/routes/files.js +0 -1
- package/dist/server/routes/files.js.map +1 -1
- package/dist/server/routes/improvise.d.ts.map +1 -1
- package/dist/server/routes/improvise.js +0 -1
- package/dist/server/routes/improvise.js.map +1 -1
- package/dist/server/routes/index.d.ts.map +1 -1
- package/dist/server/routes/index.js +0 -1
- package/dist/server/routes/index.js.map +1 -1
- package/dist/server/routes/instances.d.ts.map +1 -1
- package/dist/server/routes/instances.js +0 -1
- package/dist/server/routes/instances.js.map +1 -1
- package/dist/server/routes/notifications.d.ts.map +1 -1
- package/dist/server/routes/notifications.js +0 -1
- package/dist/server/routes/notifications.js.map +1 -1
- package/dist/server/server-setup.d.ts +16 -1
- package/dist/server/server-setup.d.ts.map +1 -1
- package/dist/server/server-setup.js +107 -1
- package/dist/server/server-setup.js.map +1 -1
- package/dist/server/services/analytics.d.ts.map +1 -1
- package/dist/server/services/analytics.js +0 -1
- package/dist/server/services/analytics.js.map +1 -1
- package/dist/server/services/auth.d.ts.map +1 -1
- package/dist/server/services/auth.js +0 -1
- package/dist/server/services/auth.js.map +1 -1
- package/dist/server/services/client-id.d.ts.map +1 -1
- package/dist/server/services/client-id.js +0 -1
- package/dist/server/services/client-id.js.map +1 -1
- package/dist/server/services/file-explorer-ops.d.ts.map +1 -1
- package/dist/server/services/file-explorer-ops.js +0 -1
- package/dist/server/services/file-explorer-ops.js.map +1 -1
- package/dist/server/services/files.d.ts.map +1 -1
- package/dist/server/services/files.js +0 -1
- package/dist/server/services/files.js.map +1 -1
- package/dist/server/services/instances.d.ts.map +1 -1
- package/dist/server/services/instances.js +0 -1
- package/dist/server/services/instances.js.map +1 -1
- package/dist/server/services/pathUtils.d.ts.map +1 -1
- package/dist/server/services/pathUtils.js +0 -1
- package/dist/server/services/pathUtils.js.map +1 -1
- package/dist/server/services/plan/agent-loader.d.ts.map +1 -1
- package/dist/server/services/plan/agent-loader.js +0 -1
- package/dist/server/services/plan/agent-loader.js.map +1 -1
- package/dist/server/services/plan/board-config.d.ts +21 -0
- package/dist/server/services/plan/board-config.d.ts.map +1 -0
- package/dist/server/services/plan/board-config.js +111 -0
- package/dist/server/services/plan/board-config.js.map +1 -0
- package/dist/server/services/plan/composer.d.ts +1 -1
- package/dist/server/services/plan/composer.d.ts.map +1 -1
- package/dist/server/services/plan/composer.js +7 -6
- package/dist/server/services/plan/composer.js.map +1 -1
- package/dist/server/services/plan/config-installer.d.ts.map +1 -1
- package/dist/server/services/plan/config-installer.js +0 -1
- package/dist/server/services/plan/config-installer.js.map +1 -1
- package/dist/server/services/plan/dependency-resolver.d.ts.map +1 -1
- package/dist/server/services/plan/dependency-resolver.js +0 -1
- package/dist/server/services/plan/dependency-resolver.js.map +1 -1
- package/dist/server/services/plan/executor.d.ts +48 -48
- package/dist/server/services/plan/executor.d.ts.map +1 -1
- package/dist/server/services/plan/executor.js +202 -458
- package/dist/server/services/plan/executor.js.map +1 -1
- package/dist/server/services/plan/front-matter.d.ts.map +1 -1
- package/dist/server/services/plan/front-matter.js +0 -1
- package/dist/server/services/plan/front-matter.js.map +1 -1
- package/dist/server/services/plan/issue-classification.d.ts.map +1 -1
- package/dist/server/services/plan/issue-classification.js +0 -1
- package/dist/server/services/plan/issue-classification.js.map +1 -1
- package/dist/server/services/plan/issue-loader.d.ts +16 -0
- package/dist/server/services/plan/issue-loader.d.ts.map +1 -0
- package/dist/server/services/plan/issue-loader.js +45 -0
- package/dist/server/services/plan/issue-loader.js.map +1 -0
- package/dist/server/services/plan/issue-prompt-builder.d.ts.map +1 -1
- package/dist/server/services/plan/issue-prompt-builder.js +0 -1
- package/dist/server/services/plan/issue-prompt-builder.js.map +1 -1
- package/dist/server/services/plan/issue-retry.d.ts +3 -1
- package/dist/server/services/plan/issue-retry.d.ts.map +1 -1
- package/dist/server/services/plan/issue-retry.js +2 -1
- package/dist/server/services/plan/issue-retry.js.map +1 -1
- package/dist/server/services/plan/issue-writer.d.ts +34 -0
- package/dist/server/services/plan/issue-writer.d.ts.map +1 -0
- package/dist/server/services/plan/issue-writer.js +109 -0
- package/dist/server/services/plan/issue-writer.js.map +1 -0
- package/dist/server/services/plan/output-manager.js +2 -2
- package/dist/server/services/plan/output-manager.js.map +1 -1
- package/dist/server/services/plan/parser-core.d.ts.map +1 -1
- package/dist/server/services/plan/parser-core.js +0 -1
- package/dist/server/services/plan/parser-core.js.map +1 -1
- package/dist/server/services/plan/parser-migration.d.ts.map +1 -1
- package/dist/server/services/plan/parser-migration.js +0 -1
- package/dist/server/services/plan/parser-migration.js.map +1 -1
- package/dist/server/services/plan/parser.d.ts.map +1 -1
- package/dist/server/services/plan/parser.js +0 -1
- package/dist/server/services/plan/parser.js.map +1 -1
- package/dist/server/services/plan/progress-log.d.ts +11 -0
- package/dist/server/services/plan/progress-log.d.ts.map +1 -0
- package/dist/server/services/plan/progress-log.js +80 -0
- package/dist/server/services/plan/progress-log.js.map +1 -0
- package/dist/server/services/plan/prompt-builder.d.ts.map +1 -1
- package/dist/server/services/plan/prompt-builder.js +48 -32
- package/dist/server/services/plan/prompt-builder.js.map +1 -1
- package/dist/server/services/plan/readiness-planner.d.ts +15 -0
- package/dist/server/services/plan/readiness-planner.d.ts.map +1 -0
- package/dist/server/services/plan/readiness-planner.js +40 -0
- package/dist/server/services/plan/readiness-planner.js.map +1 -0
- package/dist/server/services/plan/review-gate.d.ts +31 -0
- package/dist/server/services/plan/review-gate.d.ts.map +1 -1
- package/dist/server/services/plan/review-gate.js +52 -3
- package/dist/server/services/plan/review-gate.js.map +1 -1
- package/dist/server/services/plan/state-reconciler.d.ts.map +1 -1
- package/dist/server/services/plan/state-reconciler.js +0 -1
- package/dist/server/services/plan/state-reconciler.js.map +1 -1
- package/dist/server/services/plan/types.d.ts.map +1 -1
- package/dist/server/services/plan/types.js +0 -1
- package/dist/server/services/plan/types.js.map +1 -1
- package/dist/server/services/plan/watcher.d.ts.map +1 -1
- package/dist/server/services/plan/watcher.js +0 -1
- package/dist/server/services/plan/watcher.js.map +1 -1
- package/dist/server/services/platform-credentials.d.ts.map +1 -1
- package/dist/server/services/platform-credentials.js +0 -1
- package/dist/server/services/platform-credentials.js.map +1 -1
- package/dist/server/services/platform-token-lifecycle.d.ts +70 -0
- package/dist/server/services/platform-token-lifecycle.d.ts.map +1 -0
- package/dist/server/services/platform-token-lifecycle.js +156 -0
- package/dist/server/services/platform-token-lifecycle.js.map +1 -0
- package/dist/server/services/platform.d.ts +25 -4
- package/dist/server/services/platform.d.ts.map +1 -1
- package/dist/server/services/platform.js +150 -92
- package/dist/server/services/platform.js.map +1 -1
- package/dist/server/services/sentry.d.ts.map +1 -1
- package/dist/server/services/sentry.js +0 -1
- package/dist/server/services/sentry.js.map +1 -1
- package/dist/server/services/settings.d.ts.map +1 -1
- package/dist/server/services/settings.js +0 -1
- package/dist/server/services/settings.js.map +1 -1
- package/dist/server/services/terminal/pty-manager.d.ts.map +1 -1
- package/dist/server/services/terminal/pty-manager.js +0 -1
- package/dist/server/services/terminal/pty-manager.js.map +1 -1
- package/dist/server/services/terminal/pty-utils.d.ts.map +1 -1
- package/dist/server/services/terminal/pty-utils.js +0 -1
- package/dist/server/services/terminal/pty-utils.js.map +1 -1
- package/dist/server/services/websocket/autocomplete.d.ts.map +1 -1
- package/dist/server/services/websocket/autocomplete.js +0 -1
- package/dist/server/services/websocket/autocomplete.js.map +1 -1
- package/dist/server/services/websocket/file-definition-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/file-definition-handlers.js +0 -1
- package/dist/server/services/websocket/file-definition-handlers.js.map +1 -1
- package/dist/server/services/websocket/file-download-handler.d.ts +17 -0
- package/dist/server/services/websocket/file-download-handler.d.ts.map +1 -0
- package/dist/server/services/websocket/file-download-handler.js +164 -0
- package/dist/server/services/websocket/file-download-handler.js.map +1 -0
- package/dist/server/services/websocket/file-explorer-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/file-explorer-handlers.js +0 -1
- package/dist/server/services/websocket/file-explorer-handlers.js.map +1 -1
- package/dist/server/services/websocket/file-search-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/file-search-handlers.js +0 -1
- package/dist/server/services/websocket/file-search-handlers.js.map +1 -1
- package/dist/server/services/websocket/file-upload-handler.d.ts +2 -3
- package/dist/server/services/websocket/file-upload-handler.d.ts.map +1 -1
- package/dist/server/services/websocket/file-upload-handler.js +4 -7
- package/dist/server/services/websocket/file-upload-handler.js.map +1 -1
- package/dist/server/services/websocket/file-utils.d.ts.map +1 -1
- package/dist/server/services/websocket/file-utils.js +0 -1
- package/dist/server/services/websocket/file-utils.js.map +1 -1
- package/dist/server/services/websocket/git-branch-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-branch-handlers.js +0 -1
- package/dist/server/services/websocket/git-branch-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-diff-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-diff-handlers.js +0 -1
- package/dist/server/services/websocket/git-diff-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-handlers.js +58 -6
- package/dist/server/services/websocket/git-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-head-watcher.d.ts.map +1 -1
- package/dist/server/services/websocket/git-head-watcher.js +0 -1
- package/dist/server/services/websocket/git-head-watcher.js.map +1 -1
- package/dist/server/services/websocket/git-log-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-log-handlers.js +0 -1
- package/dist/server/services/websocket/git-log-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-pr-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-pr-handlers.js +0 -1
- package/dist/server/services/websocket/git-pr-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-tag-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-tag-handlers.js +0 -1
- package/dist/server/services/websocket/git-tag-handlers.js.map +1 -1
- package/dist/server/services/websocket/git-utils.d.ts +18 -3
- package/dist/server/services/websocket/git-utils.d.ts.map +1 -1
- package/dist/server/services/websocket/git-utils.js +58 -8
- package/dist/server/services/websocket/git-utils.js.map +1 -1
- package/dist/server/services/websocket/git-worktree-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-worktree-handlers.js +258 -16
- package/dist/server/services/websocket/git-worktree-handlers.js.map +1 -1
- package/dist/server/services/websocket/handler-context.d.ts +15 -0
- package/dist/server/services/websocket/handler-context.d.ts.map +1 -1
- package/dist/server/services/websocket/handler-context.js +0 -1
- package/dist/server/services/websocket/handler-context.js.map +1 -1
- package/dist/server/services/websocket/handler.d.ts +7 -0
- package/dist/server/services/websocket/handler.d.ts.map +1 -1
- package/dist/server/services/websocket/handler.js +76 -15
- package/dist/server/services/websocket/handler.js.map +1 -1
- package/dist/server/services/websocket/index.d.ts.map +1 -1
- package/dist/server/services/websocket/index.js +0 -1
- package/dist/server/services/websocket/index.js.map +1 -1
- package/dist/server/services/websocket/msg-id-tracker.d.ts +21 -0
- package/dist/server/services/websocket/msg-id-tracker.d.ts.map +1 -0
- package/dist/server/services/websocket/msg-id-tracker.js +76 -0
- package/dist/server/services/websocket/msg-id-tracker.js.map +1 -0
- package/dist/server/services/websocket/plan-board-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-board-handlers.js +0 -1
- package/dist/server/services/websocket/plan-board-handlers.js.map +1 -1
- package/dist/server/services/websocket/plan-execution-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-execution-handlers.js +6 -2
- package/dist/server/services/websocket/plan-execution-handlers.js.map +1 -1
- package/dist/server/services/websocket/plan-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-handlers.js +0 -1
- package/dist/server/services/websocket/plan-handlers.js.map +1 -1
- package/dist/server/services/websocket/plan-helpers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-helpers.js +0 -1
- package/dist/server/services/websocket/plan-helpers.js.map +1 -1
- package/dist/server/services/websocket/plan-issue-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-issue-handlers.js +0 -1
- package/dist/server/services/websocket/plan-issue-handlers.js.map +1 -1
- package/dist/server/services/websocket/plan-sprint-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/plan-sprint-handlers.js +0 -1
- package/dist/server/services/websocket/plan-sprint-handlers.js.map +1 -1
- package/dist/server/services/websocket/quality-complexity.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-complexity.js +0 -1
- package/dist/server/services/websocket/quality-complexity.js.map +1 -1
- package/dist/server/services/websocket/quality-grading.d.ts +46 -0
- package/dist/server/services/websocket/quality-grading.d.ts.map +1 -0
- package/dist/server/services/websocket/quality-grading.js +482 -0
- package/dist/server/services/websocket/quality-grading.js.map +1 -0
- package/dist/server/services/websocket/quality-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-handlers.js +15 -4
- package/dist/server/services/websocket/quality-handlers.js.map +1 -1
- package/dist/server/services/websocket/quality-linting.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-linting.js +0 -1
- package/dist/server/services/websocket/quality-linting.js.map +1 -1
- package/dist/server/services/websocket/quality-persistence.d.ts +14 -0
- package/dist/server/services/websocket/quality-persistence.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-persistence.js +28 -12
- package/dist/server/services/websocket/quality-persistence.js.map +1 -1
- package/dist/server/services/websocket/quality-review-agent.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-review-agent.js +2 -3
- package/dist/server/services/websocket/quality-review-agent.js.map +1 -1
- package/dist/server/services/websocket/quality-service.d.ts +3 -1
- package/dist/server/services/websocket/quality-service.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-service.js +53 -58
- package/dist/server/services/websocket/quality-service.js.map +1 -1
- package/dist/server/services/websocket/quality-tools.d.ts +1 -1
- package/dist/server/services/websocket/quality-tools.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-tools.js +6 -3
- package/dist/server/services/websocket/quality-tools.js.map +1 -1
- package/dist/server/services/websocket/quality-types.d.ts +18 -2
- package/dist/server/services/websocket/quality-types.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-types.js +0 -1
- package/dist/server/services/websocket/quality-types.js.map +1 -1
- package/dist/server/services/websocket/session-handlers.d.ts +48 -2
- package/dist/server/services/websocket/session-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/session-handlers.js +204 -66
- package/dist/server/services/websocket/session-handlers.js.map +1 -1
- package/dist/server/services/websocket/session-history.d.ts.map +1 -1
- package/dist/server/services/websocket/session-history.js +0 -1
- package/dist/server/services/websocket/session-history.js.map +1 -1
- package/dist/server/services/websocket/session-initialization.d.ts +2 -2
- package/dist/server/services/websocket/session-initialization.d.ts.map +1 -1
- package/dist/server/services/websocket/session-initialization.js +75 -18
- package/dist/server/services/websocket/session-initialization.js.map +1 -1
- package/dist/server/services/websocket/session-registry.d.ts +29 -1
- package/dist/server/services/websocket/session-registry.d.ts.map +1 -1
- package/dist/server/services/websocket/session-registry.js +53 -5
- package/dist/server/services/websocket/session-registry.js.map +1 -1
- package/dist/server/services/websocket/settings-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/settings-handlers.js +0 -1
- package/dist/server/services/websocket/settings-handlers.js.map +1 -1
- package/dist/server/services/websocket/skill-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/skill-handlers.js +0 -1
- package/dist/server/services/websocket/skill-handlers.js.map +1 -1
- package/dist/server/services/websocket/skill-watcher.d.ts.map +1 -1
- package/dist/server/services/websocket/skill-watcher.js +0 -1
- package/dist/server/services/websocket/skill-watcher.js.map +1 -1
- package/dist/server/services/websocket/tab-broadcast.d.ts +24 -0
- package/dist/server/services/websocket/tab-broadcast.d.ts.map +1 -0
- package/dist/server/services/websocket/tab-broadcast.js +12 -0
- package/dist/server/services/websocket/tab-broadcast.js.map +1 -0
- package/dist/server/services/websocket/tab-event-buffer.d.ts +103 -0
- package/dist/server/services/websocket/tab-event-buffer.d.ts.map +1 -0
- package/dist/server/services/websocket/tab-event-buffer.js +106 -0
- package/dist/server/services/websocket/tab-event-buffer.js.map +1 -0
- package/dist/server/services/websocket/tab-event-replay.d.ts +20 -0
- package/dist/server/services/websocket/tab-event-replay.d.ts.map +1 -0
- package/dist/server/services/websocket/tab-event-replay.js +20 -0
- package/dist/server/services/websocket/tab-event-replay.js.map +1 -0
- package/dist/server/services/websocket/tab-handlers.d.ts +0 -1
- package/dist/server/services/websocket/tab-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/tab-handlers.js +2 -10
- package/dist/server/services/websocket/tab-handlers.js.map +1 -1
- package/dist/server/services/websocket/terminal-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/terminal-handlers.js +39 -4
- package/dist/server/services/websocket/terminal-handlers.js.map +1 -1
- package/dist/server/services/websocket/types.d.ts +17 -8
- package/dist/server/services/websocket/types.d.ts.map +1 -1
- package/dist/server/services/websocket/types.js +8 -7
- package/dist/server/services/websocket/types.js.map +1 -1
- package/dist/server/utils/agent-manager.d.ts.map +1 -1
- package/dist/server/utils/agent-manager.js +0 -1
- package/dist/server/utils/agent-manager.js.map +1 -1
- package/dist/server/utils/paths.d.ts.map +1 -1
- package/dist/server/utils/paths.js +0 -1
- package/dist/server/utils/paths.js.map +1 -1
- package/dist/server/utils/port-manager.d.ts.map +1 -1
- package/dist/server/utils/port-manager.js +0 -1
- package/dist/server/utils/port-manager.js.map +1 -1
- package/dist/server/utils/port.d.ts.map +1 -1
- package/dist/server/utils/port.js +0 -1
- package/dist/server/utils/port.js.map +1 -1
- package/package.json +2 -2
- package/server/README.md +1 -1
- package/server/cli/headless/claude-invoker-process.ts +0 -1
- package/server/cli/headless/claude-invoker-stall.ts +7 -3
- package/server/cli/headless/claude-invoker-stream.ts +0 -1
- package/server/cli/headless/claude-invoker-tools.ts +0 -1
- package/server/cli/headless/claude-invoker.ts +1 -2
- package/server/cli/headless/haiku-assessments.ts +0 -1
- package/server/cli/headless/headless-logger.ts +0 -1
- package/server/cli/headless/index.ts +0 -1
- package/server/cli/headless/native-timeout-detector.ts +0 -1
- package/server/cli/headless/output-utils.ts +0 -1
- package/server/cli/headless/prompt-utils.ts +0 -1
- package/server/cli/headless/resilient-runner.ts +0 -1
- package/server/cli/headless/retry-strategies.ts +0 -1
- package/server/cli/headless/runner.ts +67 -73
- package/server/cli/headless/stall-assessor.ts +9 -5
- package/server/cli/headless/tool-watchdog.ts +0 -1
- package/server/cli/headless/types.ts +1 -2
- package/server/cli/improvisation-attachments.ts +0 -1
- package/server/cli/improvisation-history-store.ts +61 -0
- package/server/cli/improvisation-movements.ts +119 -0
- package/server/cli/improvisation-output-queue.ts +41 -0
- package/server/cli/improvisation-retry.ts +25 -601
- package/server/cli/improvisation-session-manager.ts +74 -161
- package/server/cli/improvisation-types.ts +0 -1
- package/server/cli/retry/retry-best-result.ts +69 -0
- package/server/cli/retry/retry-context-loss.ts +86 -0
- package/server/cli/retry/retry-premature-completion.ts +112 -0
- package/server/cli/retry/retry-recovery-strategies.ts +246 -0
- package/server/cli/retry/retry-resume-strategy.ts +32 -0
- package/server/cli/retry/retry-runner-factory.ts +69 -0
- package/server/cli/retry/retry-tool-results.ts +30 -0
- package/server/cli/retry/retry-types.ts +31 -0
- package/server/index.ts +37 -124
- package/server/mcp/bouncer-cli.ts +0 -1
- package/server/mcp/bouncer-haiku.ts +0 -1
- package/server/mcp/bouncer-integration.ts +0 -1
- package/server/mcp/security-analysis.ts +0 -1
- package/server/mcp/security-audit.ts +0 -1
- package/server/mcp/security-patterns.ts +0 -1
- package/server/mcp/server.ts +0 -1
- package/server/routes/files.ts +0 -1
- package/server/routes/improvise.ts +0 -1
- package/server/routes/index.ts +0 -1
- package/server/routes/instances.ts +0 -1
- package/server/routes/notifications.ts +0 -1
- package/server/server-setup.ts +126 -2
- package/server/services/analytics.ts +0 -1
- package/server/services/auth.ts +0 -1
- package/server/services/client-id.ts +0 -1
- package/server/services/file-explorer-ops.ts +0 -1
- package/server/services/files.ts +0 -1
- package/server/services/instances.ts +0 -1
- package/server/services/pathUtils.ts +0 -1
- package/server/services/plan/agent-loader.ts +0 -1
- package/server/services/plan/agents/assess-stall.md +11 -4
- package/server/services/plan/agents/code-review.md +13 -11
- package/server/services/plan/board-config.ts +121 -0
- package/server/services/plan/composer.ts +7 -6
- package/server/services/plan/config-installer.ts +0 -1
- package/server/services/plan/dependency-resolver.ts +0 -1
- package/server/services/plan/executor.ts +259 -470
- package/server/services/plan/front-matter.ts +0 -1
- package/server/services/plan/issue-classification.ts +0 -1
- package/server/services/plan/issue-loader.ts +63 -0
- package/server/services/plan/issue-prompt-builder.ts +0 -1
- package/server/services/plan/issue-retry.ts +5 -2
- package/server/services/plan/issue-writer.ts +136 -0
- package/server/services/plan/output-manager.ts +2 -2
- package/server/services/plan/parser-core.ts +0 -1
- package/server/services/plan/parser-migration.ts +0 -1
- package/server/services/plan/parser.ts +0 -1
- package/server/services/plan/progress-log.ts +91 -0
- package/server/services/plan/prompt-builder.ts +73 -36
- package/server/services/plan/readiness-planner.ts +49 -0
- package/server/services/plan/review-gate.ts +102 -3
- package/server/services/plan/state-reconciler.ts +0 -1
- package/server/services/plan/types.ts +0 -1
- package/server/services/plan/watcher.ts +0 -1
- package/server/services/platform-credentials.ts +0 -1
- package/server/services/platform-token-lifecycle.ts +171 -0
- package/server/services/platform.ts +168 -105
- package/server/services/sentry.ts +0 -1
- package/server/services/settings.ts +0 -1
- package/server/services/terminal/pty-manager.ts +0 -1
- package/server/services/terminal/pty-utils.ts +0 -1
- package/server/services/websocket/autocomplete.ts +0 -1
- package/server/services/websocket/file-definition-handlers.ts +0 -1
- package/server/services/websocket/file-download-handler.ts +190 -0
- package/server/services/websocket/file-explorer-handlers.ts +0 -1
- package/server/services/websocket/file-search-handlers.ts +0 -1
- package/server/services/websocket/file-upload-handler.ts +6 -5
- package/server/services/websocket/file-utils.ts +0 -1
- package/server/services/websocket/git-branch-handlers.ts +0 -1
- package/server/services/websocket/git-diff-handlers.ts +0 -1
- package/server/services/websocket/git-handlers.ts +66 -10
- package/server/services/websocket/git-head-watcher.ts +0 -1
- package/server/services/websocket/git-log-handlers.ts +0 -1
- package/server/services/websocket/git-pr-handlers.ts +0 -1
- package/server/services/websocket/git-tag-handlers.ts +0 -1
- package/server/services/websocket/git-utils.ts +69 -9
- package/server/services/websocket/git-worktree-handlers.ts +289 -19
- package/server/services/websocket/handler-context.ts +15 -1
- package/server/services/websocket/handler.ts +79 -16
- package/server/services/websocket/index.ts +0 -1
- package/server/services/websocket/msg-id-tracker.ts +83 -0
- package/server/services/websocket/plan-board-handlers.ts +0 -1
- package/server/services/websocket/plan-execution-handlers.ts +6 -2
- package/server/services/websocket/plan-handlers.ts +0 -1
- package/server/services/websocket/plan-helpers.ts +0 -1
- package/server/services/websocket/plan-issue-handlers.ts +0 -1
- package/server/services/websocket/plan-sprint-handlers.ts +0 -1
- package/server/services/websocket/quality-complexity.ts +0 -1
- package/server/services/websocket/quality-grading.ts +611 -0
- package/server/services/websocket/quality-handlers.ts +16 -4
- package/server/services/websocket/quality-linting.ts +0 -1
- package/server/services/websocket/quality-persistence.ts +30 -8
- package/server/services/websocket/quality-review-agent.ts +2 -3
- package/server/services/websocket/quality-service.ts +54 -55
- package/server/services/websocket/quality-tools.ts +11 -3
- package/server/services/websocket/quality-types.ts +21 -3
- package/server/services/websocket/session-handlers.ts +213 -69
- package/server/services/websocket/session-history.ts +0 -1
- package/server/services/websocket/session-initialization.ts +83 -20
- package/server/services/websocket/session-registry.ts +61 -5
- package/server/services/websocket/settings-handlers.ts +0 -1
- package/server/services/websocket/skill-handlers.ts +0 -1
- package/server/services/websocket/skill-watcher.ts +0 -1
- package/server/services/websocket/tab-broadcast.ts +37 -0
- package/server/services/websocket/tab-event-buffer.ts +158 -0
- package/server/services/websocket/tab-event-replay.ts +41 -0
- package/server/services/websocket/tab-handlers.ts +2 -10
- package/server/services/websocket/terminal-handlers.ts +39 -3
- package/server/services/websocket/types.ts +19 -7
- package/server/utils/agent-manager.ts +0 -1
- package/server/utils/paths.ts +0 -1
- package/server/utils/port-manager.ts +0 -1
- package/server/utils/port.ts +0 -1
|
@@ -0,0 +1,246 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Recovery strategies: context-loss (inter-movement / native-timeout),
|
|
5
|
+
* tool-timeout checkpoint retry, and signal-crash retry. Each `shouldRetry*`
|
|
6
|
+
* or `apply*Retry` function returns true when the retry loop should loop.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { AnalyticsEvents, trackEvent } from '../../services/analytics.js';
|
|
10
|
+
import type { ExecutionCheckpoint } from '../headless/types.js';
|
|
11
|
+
import type { HeadlessRunResult, MovementRecord, RetryLoopState } from '../improvisation-types.js';
|
|
12
|
+
import {
|
|
13
|
+
buildContextRecoveryPrompt,
|
|
14
|
+
buildFreshRecoveryPrompt,
|
|
15
|
+
buildInterMovementRecoveryPrompt,
|
|
16
|
+
buildResumeRetryPrompt,
|
|
17
|
+
buildRetryPrompt,
|
|
18
|
+
buildSignalCrashRecoveryPrompt,
|
|
19
|
+
extractHistoricalToolResults,
|
|
20
|
+
} from '../prompt-builders.js';
|
|
21
|
+
import { accumulateToolResults } from './retry-tool-results.js';
|
|
22
|
+
import type { RetryCallbacks, RetrySessionState } from './retry-types.js';
|
|
23
|
+
|
|
24
|
+
// āā Context-loss recovery āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
25
|
+
|
|
26
|
+
/** Handle inter-movement context loss recovery (resume session expired) */
|
|
27
|
+
export function applyInterMovementRecovery(
|
|
28
|
+
state: RetryLoopState,
|
|
29
|
+
promptWithAttachments: string,
|
|
30
|
+
history: MovementRecord[],
|
|
31
|
+
callbacks: RetryCallbacks,
|
|
32
|
+
): void {
|
|
33
|
+
const historicalResults = extractHistoricalToolResults(history);
|
|
34
|
+
const allResults = [...historicalResults, ...state.accumulatedToolResults];
|
|
35
|
+
|
|
36
|
+
callbacks.emit('onAutoRetry', {
|
|
37
|
+
retryNumber: state.retryNumber,
|
|
38
|
+
maxRetries: 3,
|
|
39
|
+
toolName: 'InterMovementRecovery',
|
|
40
|
+
completedCount: allResults.length,
|
|
41
|
+
});
|
|
42
|
+
callbacks.queueOutput(
|
|
43
|
+
`\n[[MSTRO_CONTEXT_RECOVERY]] Session context expired ā continuing with ${allResults.length} preserved results from prior work (retry ${state.retryNumber}/3).\n`
|
|
44
|
+
);
|
|
45
|
+
callbacks.flushOutputQueue();
|
|
46
|
+
|
|
47
|
+
state.freshRecoveryMode = true;
|
|
48
|
+
state.currentPrompt = buildInterMovementRecoveryPrompt(promptWithAttachments, allResults, history);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
/** Handle native-timeout context loss recovery (tool timeouts caused confusion) */
|
|
52
|
+
export function applyNativeTimeoutRecovery(
|
|
53
|
+
result: HeadlessRunResult,
|
|
54
|
+
state: RetryLoopState,
|
|
55
|
+
promptWithAttachments: string,
|
|
56
|
+
session: RetrySessionState,
|
|
57
|
+
callbacks: RetryCallbacks,
|
|
58
|
+
): void {
|
|
59
|
+
const completedCount = state.accumulatedToolResults.length;
|
|
60
|
+
|
|
61
|
+
callbacks.emit('onAutoRetry', {
|
|
62
|
+
retryNumber: state.retryNumber,
|
|
63
|
+
maxRetries: 3,
|
|
64
|
+
toolName: 'ContextRecovery',
|
|
65
|
+
completedCount,
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
if (result.claudeSessionId && state.retryNumber === 1) {
|
|
69
|
+
callbacks.queueOutput(
|
|
70
|
+
`\n[[MSTRO_CONTEXT_RECOVERY]] Context loss detected ā resuming session with ${completedCount} preserved results (retry ${state.retryNumber}/3).\n`
|
|
71
|
+
);
|
|
72
|
+
callbacks.flushOutputQueue();
|
|
73
|
+
state.contextRecoverySessionId = result.claudeSessionId;
|
|
74
|
+
session.claudeSessionId = result.claudeSessionId;
|
|
75
|
+
state.currentPrompt = buildContextRecoveryPrompt(promptWithAttachments);
|
|
76
|
+
} else {
|
|
77
|
+
callbacks.queueOutput(
|
|
78
|
+
`\n[[MSTRO_CONTEXT_RECOVERY]] Continuing with fresh context ā ${completedCount} preserved results injected (retry ${state.retryNumber}/3).\n`
|
|
79
|
+
);
|
|
80
|
+
callbacks.flushOutputQueue();
|
|
81
|
+
state.freshRecoveryMode = true;
|
|
82
|
+
state.currentPrompt = buildFreshRecoveryPrompt(promptWithAttachments, state.accumulatedToolResults, state.timedOutTools);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/** Check if context loss recovery should trigger. Returns true if loop should continue. */
|
|
87
|
+
export function shouldRetryContextLoss(
|
|
88
|
+
result: HeadlessRunResult,
|
|
89
|
+
state: RetryLoopState,
|
|
90
|
+
session: RetrySessionState,
|
|
91
|
+
useResume: boolean,
|
|
92
|
+
nativeTimeouts: number,
|
|
93
|
+
maxRetries: number,
|
|
94
|
+
promptWithAttachments: string,
|
|
95
|
+
callbacks: RetryCallbacks,
|
|
96
|
+
): boolean {
|
|
97
|
+
if (state.checkpointRef.value || state.retryNumber >= maxRetries || !state.contextLost) {
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
accumulateToolResults(result, state);
|
|
101
|
+
state.retryNumber++;
|
|
102
|
+
const path = (useResume && nativeTimeouts === 0) ? 'InterMovementRecovery' : 'NativeTimeoutRecovery';
|
|
103
|
+
state.retryLog.push({
|
|
104
|
+
retryNumber: state.retryNumber,
|
|
105
|
+
path,
|
|
106
|
+
reason: `Context lost (${nativeTimeouts} timeouts, ${state.accumulatedToolResults.length} tools preserved)`,
|
|
107
|
+
timestamp: Date.now(),
|
|
108
|
+
});
|
|
109
|
+
if (useResume && nativeTimeouts === 0) {
|
|
110
|
+
applyInterMovementRecovery(state, promptWithAttachments, session.history.movements, callbacks);
|
|
111
|
+
} else {
|
|
112
|
+
applyNativeTimeoutRecovery(result, state, promptWithAttachments, session, callbacks);
|
|
113
|
+
}
|
|
114
|
+
return true;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// āā Tool-timeout retry āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
118
|
+
|
|
119
|
+
/** Handle tool timeout checkpoint. Returns true if loop should continue. */
|
|
120
|
+
export function applyToolTimeoutRetry(
|
|
121
|
+
state: RetryLoopState,
|
|
122
|
+
maxRetries: number,
|
|
123
|
+
promptWithAttachments: string,
|
|
124
|
+
callbacks: RetryCallbacks,
|
|
125
|
+
model: string | undefined,
|
|
126
|
+
effortLevel: string | undefined,
|
|
127
|
+
): boolean {
|
|
128
|
+
if (!state.checkpointRef.value || state.retryNumber >= maxRetries) {
|
|
129
|
+
return false;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
const cp: ExecutionCheckpoint = state.checkpointRef.value;
|
|
133
|
+
state.retryNumber++;
|
|
134
|
+
|
|
135
|
+
state.timedOutTools.push({
|
|
136
|
+
toolName: cp.hungTool.toolName,
|
|
137
|
+
input: cp.hungTool.input ?? {},
|
|
138
|
+
timeoutMs: cp.hungTool.timeoutMs,
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
const canResumeSession = cp.inProgressTools.length === 0 && !!cp.claudeSessionId;
|
|
142
|
+
state.retryLog.push({
|
|
143
|
+
retryNumber: state.retryNumber,
|
|
144
|
+
path: 'ToolTimeout',
|
|
145
|
+
reason: `${cp.hungTool.toolName} timed out after ${cp.hungTool.timeoutMs}ms, ${cp.completedTools.length} tools completed, ${canResumeSession ? 'resuming' : 'fresh start'}`,
|
|
146
|
+
timestamp: Date.now(),
|
|
147
|
+
});
|
|
148
|
+
callbacks.emit('onAutoRetry', {
|
|
149
|
+
retryNumber: state.retryNumber,
|
|
150
|
+
maxRetries,
|
|
151
|
+
toolName: cp.hungTool.toolName,
|
|
152
|
+
url: cp.hungTool.url,
|
|
153
|
+
completedCount: cp.completedTools.length,
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
trackEvent(AnalyticsEvents.IMPROVISE_AUTO_RETRY, {
|
|
157
|
+
retry_number: state.retryNumber,
|
|
158
|
+
hung_tool: cp.hungTool.toolName,
|
|
159
|
+
hung_url: cp.hungTool.url?.slice(0, 200),
|
|
160
|
+
completed_tools: cp.completedTools.length,
|
|
161
|
+
elapsed_ms: cp.elapsedMs,
|
|
162
|
+
resume_attempted: canResumeSession,
|
|
163
|
+
model: model || 'default',
|
|
164
|
+
effort_level: effortLevel || 'auto',
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
state.currentPrompt = canResumeSession
|
|
168
|
+
? buildResumeRetryPrompt(cp, state.timedOutTools)
|
|
169
|
+
: buildRetryPrompt(cp, promptWithAttachments, state.timedOutTools);
|
|
170
|
+
|
|
171
|
+
callbacks.queueOutput(
|
|
172
|
+
`\n[[MSTRO_AUTO_RETRY]] Auto-retry ${state.retryNumber}/${maxRetries}: ${canResumeSession ? 'Resuming session' : 'Continuing'} with ${cp.completedTools.length} successful results, skipping failed ${cp.hungTool.toolName}.\n`
|
|
173
|
+
);
|
|
174
|
+
callbacks.flushOutputQueue();
|
|
175
|
+
|
|
176
|
+
return true;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
// āā Signal-crash retry āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
180
|
+
|
|
181
|
+
/** Detect and retry after a signal crash. Returns true if loop should continue. */
|
|
182
|
+
export function shouldRetrySignalCrash(
|
|
183
|
+
result: HeadlessRunResult,
|
|
184
|
+
state: RetryLoopState,
|
|
185
|
+
session: RetrySessionState,
|
|
186
|
+
maxRetries: number,
|
|
187
|
+
promptWithAttachments: string,
|
|
188
|
+
callbacks: RetryCallbacks,
|
|
189
|
+
): boolean {
|
|
190
|
+
const isSignalCrash = !!result.signalName;
|
|
191
|
+
const exitCodeSignal = !result.completed && !result.signalName && result.error?.match(/exited with code (1[2-9]\d|[2-9]\d{2})/);
|
|
192
|
+
if ((!isSignalCrash && !exitCodeSignal) || state.retryNumber >= maxRetries) {
|
|
193
|
+
return false;
|
|
194
|
+
}
|
|
195
|
+
if (state.checkpointRef.value) {
|
|
196
|
+
return false;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
accumulateToolResults(result, state);
|
|
200
|
+
state.retryNumber++;
|
|
201
|
+
|
|
202
|
+
const completedCount = state.accumulatedToolResults.length;
|
|
203
|
+
const signalInfo = result.signalName || 'unknown signal';
|
|
204
|
+
const useResume = !!result.claudeSessionId && state.retryNumber === 1;
|
|
205
|
+
|
|
206
|
+
state.retryLog.push({
|
|
207
|
+
retryNumber: state.retryNumber,
|
|
208
|
+
path: 'SignalCrash',
|
|
209
|
+
reason: `Process killed (${signalInfo}), ${completedCount} tools preserved, ${useResume ? 'resuming' : 'fresh start'}`,
|
|
210
|
+
timestamp: Date.now(),
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
callbacks.emit('onAutoRetry', {
|
|
214
|
+
retryNumber: state.retryNumber,
|
|
215
|
+
maxRetries,
|
|
216
|
+
toolName: `SignalCrash(${signalInfo})`,
|
|
217
|
+
completedCount,
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
trackEvent(AnalyticsEvents.IMPROVISE_AUTO_RETRY, {
|
|
221
|
+
retry_number: state.retryNumber,
|
|
222
|
+
hung_tool: `signal_crash:${signalInfo}`,
|
|
223
|
+
completed_tools: completedCount,
|
|
224
|
+
resume_attempted: useResume,
|
|
225
|
+
});
|
|
226
|
+
|
|
227
|
+
if (useResume) {
|
|
228
|
+
callbacks.queueOutput(
|
|
229
|
+
`\n[[MSTRO_SIGNAL_RECOVERY]] Process killed (${signalInfo}) ā resuming session with ${completedCount} preserved results (retry ${state.retryNumber}/${maxRetries}).\n`
|
|
230
|
+
);
|
|
231
|
+
callbacks.flushOutputQueue();
|
|
232
|
+
state.contextRecoverySessionId = result.claudeSessionId;
|
|
233
|
+
session.claudeSessionId = result.claudeSessionId;
|
|
234
|
+
state.currentPrompt = buildSignalCrashRecoveryPrompt(promptWithAttachments, true);
|
|
235
|
+
} else {
|
|
236
|
+
callbacks.queueOutput(
|
|
237
|
+
`\n[[MSTRO_SIGNAL_RECOVERY]] Process killed (${signalInfo}) ā restarting with ${completedCount} preserved results (retry ${state.retryNumber}/${maxRetries}).\n`
|
|
238
|
+
);
|
|
239
|
+
callbacks.flushOutputQueue();
|
|
240
|
+
state.freshRecoveryMode = true;
|
|
241
|
+
const allResults = [...extractHistoricalToolResults(session.history.movements), ...state.accumulatedToolResults];
|
|
242
|
+
state.currentPrompt = buildSignalCrashRecoveryPrompt(promptWithAttachments, false, allResults);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
return true;
|
|
246
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Picks how the next retry iteration should resume execution:
|
|
5
|
+
* whether to pass `--resume`, and which Claude session ID (if any) to use.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { RetryLoopState } from '../improvisation-types.js';
|
|
9
|
+
import type { RetrySessionState } from './retry-types.js';
|
|
10
|
+
|
|
11
|
+
/** Determine whether to use --resume and which session ID */
|
|
12
|
+
export function determineResumeStrategy(
|
|
13
|
+
state: RetryLoopState,
|
|
14
|
+
session: RetrySessionState,
|
|
15
|
+
): { useResume: boolean; resumeSessionId: string | undefined } {
|
|
16
|
+
if (state.freshRecoveryMode) {
|
|
17
|
+
state.freshRecoveryMode = false;
|
|
18
|
+
return { useResume: false, resumeSessionId: undefined };
|
|
19
|
+
}
|
|
20
|
+
if (state.contextRecoverySessionId) {
|
|
21
|
+
const id = state.contextRecoverySessionId;
|
|
22
|
+
state.contextRecoverySessionId = undefined;
|
|
23
|
+
return { useResume: true, resumeSessionId: id };
|
|
24
|
+
}
|
|
25
|
+
if (state.retryNumber === 0) {
|
|
26
|
+
return { useResume: !session.isFirstPrompt, resumeSessionId: session.claudeSessionId };
|
|
27
|
+
}
|
|
28
|
+
if (state.lastWatchdogCheckpoint?.inProgressTools.length === 0 && state.lastWatchdogCheckpoint.claudeSessionId) {
|
|
29
|
+
return { useResume: true, resumeSessionId: state.lastWatchdogCheckpoint.claudeSessionId };
|
|
30
|
+
}
|
|
31
|
+
return { useResume: false, resumeSessionId: undefined };
|
|
32
|
+
}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Factory for the HeadlessRunner used in a single retry iteration.
|
|
5
|
+
* Wires session options into HeadlessRunner callbacks that respect the
|
|
6
|
+
* shared cancel flag and forward streaming events via `callbacks`.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { HeadlessRunner } from '../headless/index.js';
|
|
10
|
+
import type { ExecutionCheckpoint } from '../headless/types.js';
|
|
11
|
+
import type { FileAttachment, RetryLoopState } from '../improvisation-types.js';
|
|
12
|
+
import { buildHistoricalContext } from '../prompt-builders.js';
|
|
13
|
+
import type { RetryCallbacks, RetrySessionState } from './retry-types.js';
|
|
14
|
+
|
|
15
|
+
/** Create HeadlessRunner for one retry iteration */
|
|
16
|
+
export function createExecutionRunner(
|
|
17
|
+
state: RetryLoopState,
|
|
18
|
+
session: RetrySessionState,
|
|
19
|
+
callbacks: RetryCallbacks,
|
|
20
|
+
sequenceNumber: number,
|
|
21
|
+
useResume: boolean,
|
|
22
|
+
resumeSessionId: string | undefined,
|
|
23
|
+
imageAttachments: FileAttachment[] | undefined,
|
|
24
|
+
workingDirOverride?: string,
|
|
25
|
+
): HeadlessRunner {
|
|
26
|
+
return new HeadlessRunner({
|
|
27
|
+
workingDir: workingDirOverride || session.options.workingDir,
|
|
28
|
+
tokenBudgetThreshold: session.options.tokenBudgetThreshold,
|
|
29
|
+
maxSessions: session.options.maxSessions,
|
|
30
|
+
verbose: session.options.verbose,
|
|
31
|
+
noColor: session.options.noColor,
|
|
32
|
+
model: session.options.model,
|
|
33
|
+
effortLevel: session.options.effortLevel,
|
|
34
|
+
improvisationMode: true,
|
|
35
|
+
movementNumber: sequenceNumber,
|
|
36
|
+
continueSession: useResume,
|
|
37
|
+
claudeSessionId: resumeSessionId,
|
|
38
|
+
outputCallback: (text: string) => {
|
|
39
|
+
if (callbacks.isCancelled()) return;
|
|
40
|
+
callbacks.addEventLog({ type: 'output', data: { text, timestamp: Date.now() }, timestamp: Date.now() });
|
|
41
|
+
callbacks.queueOutput(text);
|
|
42
|
+
callbacks.flushOutputQueue();
|
|
43
|
+
},
|
|
44
|
+
thinkingCallback: (text: string) => {
|
|
45
|
+
if (callbacks.isCancelled()) return;
|
|
46
|
+
callbacks.addEventLog({ type: 'thinking', data: { text }, timestamp: Date.now() });
|
|
47
|
+
callbacks.emit('onThinking', text);
|
|
48
|
+
callbacks.flushOutputQueue();
|
|
49
|
+
},
|
|
50
|
+
toolUseCallback: (event) => {
|
|
51
|
+
if (callbacks.isCancelled()) return;
|
|
52
|
+
callbacks.addEventLog({ type: 'toolUse', data: { ...event, timestamp: Date.now() }, timestamp: Date.now() });
|
|
53
|
+
callbacks.emit('onToolUse', event);
|
|
54
|
+
callbacks.flushOutputQueue();
|
|
55
|
+
},
|
|
56
|
+
tokenUsageCallback: (usage) => {
|
|
57
|
+
if (callbacks.isCancelled()) return;
|
|
58
|
+
callbacks.emit('onTokenUsage', usage);
|
|
59
|
+
},
|
|
60
|
+
directPrompt: state.currentPrompt,
|
|
61
|
+
imageAttachments,
|
|
62
|
+
promptContext: (state.retryNumber === 0 && session.isResumedSession && session.isFirstPrompt)
|
|
63
|
+
? { accumulatedKnowledge: buildHistoricalContext(session.history.movements), filesModified: [] }
|
|
64
|
+
: undefined,
|
|
65
|
+
onToolTimeout: (checkpoint: ExecutionCheckpoint) => {
|
|
66
|
+
state.checkpointRef.value = checkpoint;
|
|
67
|
+
},
|
|
68
|
+
});
|
|
69
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Running buffer of tool results preserved across retries so recovery
|
|
5
|
+
* prompts can re-inject completed work instead of redoing it.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { HeadlessRunResult, RetryLoopState } from '../improvisation-types.js';
|
|
9
|
+
|
|
10
|
+
export const MAX_ACCUMULATED_RESULTS = 50;
|
|
11
|
+
|
|
12
|
+
/** Accumulate completed tool results from a run into the retry state */
|
|
13
|
+
export function accumulateToolResults(result: HeadlessRunResult, state: RetryLoopState): void {
|
|
14
|
+
if (!result.toolUseHistory) return;
|
|
15
|
+
for (const t of result.toolUseHistory) {
|
|
16
|
+
if (t.result !== undefined) {
|
|
17
|
+
state.accumulatedToolResults.push({
|
|
18
|
+
toolName: t.toolName,
|
|
19
|
+
toolId: t.toolId,
|
|
20
|
+
toolInput: t.toolInput,
|
|
21
|
+
result: t.result,
|
|
22
|
+
isError: t.isError,
|
|
23
|
+
duration: t.duration,
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
if (state.accumulatedToolResults.length > MAX_ACCUMULATED_RESULTS) {
|
|
28
|
+
state.accumulatedToolResults = state.accumulatedToolResults.slice(-MAX_ACCUMULATED_RESULTS);
|
|
29
|
+
}
|
|
30
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Shared types used across the retry modules. Defined separately so each
|
|
5
|
+
* strategy file can import them without creating circular dependencies.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import type { HeadlessRunner } from '../headless/index.js';
|
|
9
|
+
import type { ImprovisationOptions, SessionHistory } from '../improvisation-types.js';
|
|
10
|
+
|
|
11
|
+
/** Callbacks the retry logic needs from the session manager */
|
|
12
|
+
export interface RetryCallbacks {
|
|
13
|
+
isCancelled: () => boolean;
|
|
14
|
+
queueOutput: (text: string) => void;
|
|
15
|
+
flushOutputQueue: () => void;
|
|
16
|
+
emit: (event: string, ...args: unknown[]) => void;
|
|
17
|
+
addEventLog: (entry: { type: string; data: unknown; timestamp: number }) => void;
|
|
18
|
+
setRunner: (runner: HeadlessRunner | null) => void;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/** Session state the retry logic reads/writes */
|
|
22
|
+
export interface RetrySessionState {
|
|
23
|
+
options: ImprovisationOptions;
|
|
24
|
+
claudeSessionId: string | undefined;
|
|
25
|
+
isFirstPrompt: boolean;
|
|
26
|
+
isResumedSession: boolean;
|
|
27
|
+
history: SessionHistory;
|
|
28
|
+
executionStartTimestamp: number | undefined;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export const MAX_RETRIES = 3;
|
package/server/index.ts
CHANGED
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
-
// Licensed under the MIT License. See LICENSE file for details.
|
|
3
2
|
|
|
4
3
|
/**
|
|
5
4
|
* Mstro Server (Node.js + Hono)
|
|
@@ -9,14 +8,14 @@
|
|
|
9
8
|
|
|
10
9
|
import { randomBytes } from 'node:crypto'
|
|
11
10
|
import { readFileSync } from 'node:fs'
|
|
12
|
-
import type {
|
|
11
|
+
import type { Server } from 'node:http'
|
|
13
12
|
import { homedir } from 'node:os'
|
|
14
13
|
import { basename, join } from 'node:path'
|
|
15
14
|
import { serve } from '@hono/node-server'
|
|
16
15
|
import { type Context, Hono, type Next } from 'hono'
|
|
17
16
|
import { cors } from 'hono/cors'
|
|
18
17
|
import { logger } from 'hono/logger'
|
|
19
|
-
import {
|
|
18
|
+
import { WebSocketServer } from 'ws'
|
|
20
19
|
import {
|
|
21
20
|
createFileRoutes,
|
|
22
21
|
createImproviseRoutes,
|
|
@@ -24,16 +23,20 @@ import {
|
|
|
24
23
|
createNotificationRoutes,
|
|
25
24
|
createShutdownRoute
|
|
26
25
|
} from './routes/index.js'
|
|
27
|
-
import {
|
|
26
|
+
import {
|
|
27
|
+
attachLocalWebSocketRouting,
|
|
28
|
+
createPlatformRelay,
|
|
29
|
+
ensureClaudeSettings,
|
|
30
|
+
registerProcessErrorHandlers,
|
|
31
|
+
setTerminalTitle
|
|
32
|
+
} from './server-setup.js'
|
|
28
33
|
import { AnalyticsEvents, initAnalytics, shutdownAnalytics, trackEvent } from './services/analytics.js'
|
|
29
34
|
import { AuthService } from './services/auth.js'
|
|
30
35
|
import { FileService } from './services/files.js'
|
|
31
36
|
import { InstanceRegistry, type MstroInstance } from './services/instances.js'
|
|
32
|
-
import { PlatformConnection } from './services/platform.js'
|
|
33
37
|
import { captureException, flushSentry, initSentry } from './services/sentry.js'
|
|
34
38
|
import { getPTYManager, reloadPty } from './services/terminal/pty-manager.js'
|
|
35
39
|
import { WebSocketImproviseHandler } from './services/websocket/index.js'
|
|
36
|
-
import type { WSContext } from './services/websocket/types.js'
|
|
37
40
|
import { findAvailablePort } from './utils/port.js'
|
|
38
41
|
|
|
39
42
|
// Configuration
|
|
@@ -129,136 +132,46 @@ app.onError((err, c) => {
|
|
|
129
132
|
|
|
130
133
|
// āā Server Startup āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
131
134
|
|
|
132
|
-
|
|
133
|
-
initSentry()
|
|
134
|
-
await initAnalytics()
|
|
135
|
-
|
|
136
|
-
const PORT = await findAvailablePort(REQUESTED_PORT, 20)
|
|
137
|
-
_currentInstance = instanceRegistry.register(PORT, WORKING_DIR)
|
|
138
|
-
|
|
139
|
-
const server = serve({ fetch: app.fetch, port: PORT })
|
|
140
|
-
const wss = new WebSocketServer({ server: server as Server })
|
|
141
|
-
|
|
142
|
-
wss.on('connection', (ws: NodeWebSocket, req: IncomingMessage) => {
|
|
143
|
-
const url = new URL(req.url || '/', `http://localhost:${PORT}`)
|
|
144
|
-
if (url.pathname !== '/ws') {
|
|
145
|
-
ws.close(1008, 'Invalid WebSocket path')
|
|
146
|
-
return
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
const wsToken = url.searchParams.get('token')
|
|
150
|
-
if (!wsToken || !authService.validateLocalToken(wsToken)) {
|
|
151
|
-
ws.close(4001, 'Unauthorized')
|
|
152
|
-
return
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
const workingDir = WORKING_DIR
|
|
156
|
-
const wrappedWs = wrapWebSocket(ws, workingDir)
|
|
157
|
-
wsHandler.handleConnection(wrappedWs, workingDir)
|
|
158
|
-
|
|
159
|
-
ws.on('message', (data: Buffer | string) => {
|
|
160
|
-
let message = typeof data === 'string' ? data : data.toString('utf-8')
|
|
161
|
-
// Strip _permission from local WebSocket messages ā only the platform relay
|
|
162
|
-
// should inject permission metadata. Local connections are always the machine owner.
|
|
163
|
-
if (message.includes('_permission')) {
|
|
164
|
-
try {
|
|
165
|
-
const parsed = JSON.parse(message)
|
|
166
|
-
if ('_permission' in parsed) {
|
|
167
|
-
delete parsed._permission
|
|
168
|
-
message = JSON.stringify(parsed)
|
|
169
|
-
}
|
|
170
|
-
} catch { /* not JSON ā pass through */ }
|
|
171
|
-
}
|
|
172
|
-
wsHandler.handleMessage(wrappedWs, message, workingDir)
|
|
173
|
-
})
|
|
174
|
-
ws.on('close', () => wsHandler.handleClose(wrappedWs))
|
|
175
|
-
ws.on('error', (error: Error) => {
|
|
176
|
-
console.error('[WebSocket] Error:', error)
|
|
177
|
-
captureException(error, { context: 'websocket.connection' })
|
|
178
|
-
})
|
|
179
|
-
})
|
|
180
|
-
|
|
135
|
+
function logStartupBanner(port: number): void {
|
|
181
136
|
const home = homedir()
|
|
182
137
|
const displayDir = WORKING_DIR.startsWith(home) ? `~${WORKING_DIR.slice(home.length)}` : WORKING_DIR
|
|
183
138
|
console.log(`App: ${displayDir}`)
|
|
184
|
-
trackEvent(AnalyticsEvents.SERVER_STARTED, { port
|
|
185
|
-
|
|
186
|
-
// Platform relay
|
|
187
|
-
let platformRelayContext: WSContext | null = null
|
|
188
|
-
let pendingRelayMessages: unknown[] = []
|
|
189
|
-
|
|
190
|
-
const platformConnection = new PlatformConnection(WORKING_DIR, {
|
|
191
|
-
onConnected: () => {
|
|
192
|
-
console.log(`Connected: https://mstro.app`)
|
|
193
|
-
wsHandler.setUsageReporter((report) => {
|
|
194
|
-
platformConnection.send({ type: 'reportUsage', data: report })
|
|
195
|
-
})
|
|
196
|
-
},
|
|
197
|
-
onDisconnected: () => {
|
|
198
|
-
if (platformRelayContext) {
|
|
199
|
-
wsHandler.handleClose(platformRelayContext)
|
|
200
|
-
platformRelayContext = null
|
|
201
|
-
}
|
|
202
|
-
pendingRelayMessages = []
|
|
203
|
-
},
|
|
204
|
-
onWebConnected: () => {
|
|
205
|
-
// Clean up previous relay context to prevent duplicate broadcasts
|
|
206
|
-
if (platformRelayContext) {
|
|
207
|
-
wsHandler.handleClose(platformRelayContext)
|
|
208
|
-
}
|
|
209
|
-
platformRelayContext = createPlatformRelayContext(
|
|
210
|
-
(message) => platformConnection.send(message),
|
|
211
|
-
WORKING_DIR
|
|
212
|
-
)
|
|
213
|
-
wsHandler.handleConnection(platformRelayContext, WORKING_DIR)
|
|
214
|
-
if (pendingRelayMessages.length > 0) {
|
|
215
|
-
for (const message of pendingRelayMessages) {
|
|
216
|
-
wsHandler.handleMessage(platformRelayContext, JSON.stringify(message), WORKING_DIR)
|
|
217
|
-
}
|
|
218
|
-
pendingRelayMessages = []
|
|
219
|
-
}
|
|
220
|
-
},
|
|
221
|
-
onWebDisconnected: () => {
|
|
222
|
-
if (platformRelayContext) {
|
|
223
|
-
wsHandler.handleClose(platformRelayContext)
|
|
224
|
-
platformRelayContext = null
|
|
225
|
-
}
|
|
226
|
-
pendingRelayMessages = []
|
|
227
|
-
},
|
|
228
|
-
onRelayedMessage: (message) => {
|
|
229
|
-
if (platformRelayContext) {
|
|
230
|
-
wsHandler.handleMessage(platformRelayContext, JSON.stringify(message), WORKING_DIR)
|
|
231
|
-
} else {
|
|
232
|
-
// Cap pending messages to prevent unbounded memory growth while disconnected
|
|
233
|
-
if (pendingRelayMessages.length < 100) {
|
|
234
|
-
pendingRelayMessages.push(message)
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
})
|
|
239
|
-
platformConnection.connect()
|
|
240
|
-
|
|
241
|
-
// Process-level error handling
|
|
242
|
-
process.on('uncaughtException', (err) => {
|
|
243
|
-
console.error('[Server] Uncaught exception:', err)
|
|
244
|
-
captureException(err, { context: 'uncaughtException' })
|
|
245
|
-
})
|
|
246
|
-
process.on('unhandledRejection', (reason) => {
|
|
247
|
-
console.error('[Server] Unhandled rejection:', reason)
|
|
248
|
-
captureException(reason instanceof Error ? reason : new Error(String(reason)), { context: 'unhandledRejection' })
|
|
249
|
-
})
|
|
139
|
+
trackEvent(AnalyticsEvents.SERVER_STARTED, { port, working_dir_basename: basename(WORKING_DIR) })
|
|
140
|
+
}
|
|
250
141
|
|
|
251
|
-
|
|
142
|
+
function makeGracefulShutdown(deps: {
|
|
143
|
+
platformConnection: { disconnect: () => void }
|
|
144
|
+
wss: WebSocketServer
|
|
145
|
+
}) {
|
|
146
|
+
return async () => {
|
|
252
147
|
trackEvent(AnalyticsEvents.SERVER_STOPPED)
|
|
253
148
|
await Promise.all([shutdownAnalytics(), flushSentry()])
|
|
254
|
-
platformConnection.disconnect()
|
|
149
|
+
deps.platformConnection.disconnect()
|
|
255
150
|
instanceRegistry.unregister()
|
|
256
151
|
getPTYManager().closeAll()
|
|
257
|
-
wss.close()
|
|
152
|
+
deps.wss.close()
|
|
258
153
|
console.log('\n\nš Shutting down gracefully...\n')
|
|
259
154
|
process.exit(0)
|
|
260
155
|
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
async function startServer() {
|
|
159
|
+
initSentry()
|
|
160
|
+
await initAnalytics()
|
|
161
|
+
|
|
162
|
+
const port = await findAvailablePort(REQUESTED_PORT, 20)
|
|
163
|
+
_currentInstance = instanceRegistry.register(port, WORKING_DIR)
|
|
164
|
+
|
|
165
|
+
const server = serve({ fetch: app.fetch, port })
|
|
166
|
+
const wss = new WebSocketServer({ server: server as Server })
|
|
167
|
+
|
|
168
|
+
attachLocalWebSocketRouting({ wss, port, workingDir: WORKING_DIR, authService, wsHandler })
|
|
169
|
+
logStartupBanner(port)
|
|
170
|
+
|
|
171
|
+
const platformConnection = createPlatformRelay(WORKING_DIR, wsHandler)
|
|
172
|
+
registerProcessErrorHandlers()
|
|
261
173
|
|
|
174
|
+
const gracefulShutdown = makeGracefulShutdown({ platformConnection, wss })
|
|
262
175
|
process.on('SIGINT', gracefulShutdown)
|
|
263
176
|
process.on('SIGTERM', gracefulShutdown)
|
|
264
177
|
}
|
package/server/mcp/server.ts
CHANGED