mstro-app 0.5.1 → 0.5.6
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/PRIVACY.md +9 -9
- package/README.md +71 -28
- package/bin/commands/config.js +1 -1
- package/bin/mstro.js +55 -4
- package/dist/server/cli/eta-estimator.d.ts +55 -0
- package/dist/server/cli/eta-estimator.d.ts.map +1 -0
- package/dist/server/cli/eta-estimator.js +222 -0
- package/dist/server/cli/eta-estimator.js.map +1 -0
- package/dist/server/cli/headless/claude-invoker-process.d.ts.map +1 -1
- package/dist/server/cli/headless/claude-invoker-process.js +9 -1
- package/dist/server/cli/headless/claude-invoker-process.js.map +1 -1
- package/dist/server/cli/headless/mcp-config.d.ts +22 -5
- package/dist/server/cli/headless/mcp-config.d.ts.map +1 -1
- package/dist/server/cli/headless/mcp-config.js +7 -5
- package/dist/server/cli/headless/mcp-config.js.map +1 -1
- package/dist/server/cli/headless/runner.d.ts.map +1 -1
- package/dist/server/cli/headless/runner.js +19 -0
- package/dist/server/cli/headless/runner.js.map +1 -1
- package/dist/server/cli/headless/stall-assessor.d.ts +50 -0
- package/dist/server/cli/headless/stall-assessor.d.ts.map +1 -1
- package/dist/server/cli/headless/stall-assessor.js +64 -9
- package/dist/server/cli/headless/stall-assessor.js.map +1 -1
- package/dist/server/cli/headless/tool-watchdog.d.ts +21 -0
- package/dist/server/cli/headless/tool-watchdog.d.ts.map +1 -1
- package/dist/server/cli/headless/tool-watchdog.js +19 -12
- package/dist/server/cli/headless/tool-watchdog.js.map +1 -1
- package/dist/server/cli/headless/types.d.ts +16 -1
- package/dist/server/cli/headless/types.d.ts.map +1 -1
- package/dist/server/cli/improvisation-history-store.d.ts.map +1 -1
- package/dist/server/cli/improvisation-history-store.js +5 -1
- package/dist/server/cli/improvisation-history-store.js.map +1 -1
- package/dist/server/cli/improvisation-output-queue.d.ts +5 -1
- package/dist/server/cli/improvisation-output-queue.d.ts.map +1 -1
- package/dist/server/cli/improvisation-output-queue.js +30 -7
- package/dist/server/cli/improvisation-output-queue.js.map +1 -1
- package/dist/server/cli/improvisation-session-manager.d.ts +35 -0
- package/dist/server/cli/improvisation-session-manager.d.ts.map +1 -1
- package/dist/server/cli/improvisation-session-manager.js +58 -1
- package/dist/server/cli/improvisation-session-manager.js.map +1 -1
- package/dist/server/cli/improvisation-types.d.ts +9 -0
- package/dist/server/cli/improvisation-types.d.ts.map +1 -1
- package/dist/server/cli/improvisation-types.js.map +1 -1
- package/dist/server/cli/retry/retry-runner-factory.d.ts.map +1 -1
- package/dist/server/cli/retry/retry-runner-factory.js +1 -0
- package/dist/server/cli/retry/retry-runner-factory.js.map +1 -1
- package/dist/server/engines/EngineEvent.d.ts +126 -0
- package/dist/server/engines/EngineEvent.d.ts.map +1 -0
- package/dist/server/engines/EngineEvent.js +11 -0
- package/dist/server/engines/EngineEvent.js.map +1 -0
- package/dist/server/engines/claude/ClaudeCodeEngine.d.ts +47 -0
- package/dist/server/engines/claude/ClaudeCodeEngine.d.ts.map +1 -0
- package/dist/server/engines/claude/ClaudeCodeEngine.js +338 -0
- package/dist/server/engines/claude/ClaudeCodeEngine.js.map +1 -0
- package/dist/server/engines/factory.d.ts +21 -0
- package/dist/server/engines/factory.d.ts.map +1 -0
- package/dist/server/engines/factory.js +152 -0
- package/dist/server/engines/factory.js.map +1 -0
- package/dist/server/engines/opencode/OpenCodeEngine.d.ts +148 -0
- package/dist/server/engines/opencode/OpenCodeEngine.d.ts.map +1 -0
- package/dist/server/engines/opencode/OpenCodeEngine.js +630 -0
- package/dist/server/engines/opencode/OpenCodeEngine.js.map +1 -0
- package/dist/server/engines/opencode/OpenCodeServerManager.d.ts +172 -0
- package/dist/server/engines/opencode/OpenCodeServerManager.d.ts.map +1 -0
- package/dist/server/engines/opencode/OpenCodeServerManager.js +390 -0
- package/dist/server/engines/opencode/OpenCodeServerManager.js.map +1 -0
- package/dist/server/engines/opencode/model-catalog.d.ts +94 -0
- package/dist/server/engines/opencode/model-catalog.d.ts.map +1 -0
- package/dist/server/engines/opencode/model-catalog.js +141 -0
- package/dist/server/engines/opencode/model-catalog.js.map +1 -0
- package/dist/server/engines/types.d.ts +146 -0
- package/dist/server/engines/types.d.ts.map +1 -0
- package/dist/server/engines/types.js +4 -0
- package/dist/server/engines/types.js.map +1 -0
- package/dist/server/index.js +9 -2
- package/dist/server/index.js.map +1 -1
- package/dist/server/mcp/bouncer-haiku.d.ts +17 -4
- package/dist/server/mcp/bouncer-haiku.d.ts.map +1 -1
- package/dist/server/mcp/bouncer-haiku.js +8 -124
- package/dist/server/mcp/bouncer-haiku.js.map +1 -1
- package/dist/server/mcp/bouncer-integration.d.ts +45 -0
- package/dist/server/mcp/bouncer-integration.d.ts.map +1 -1
- package/dist/server/mcp/bouncer-integration.js +69 -5
- package/dist/server/mcp/bouncer-integration.js.map +1 -1
- package/dist/server/mcp/classifier/BouncerClassifier.d.ts +34 -0
- package/dist/server/mcp/classifier/BouncerClassifier.d.ts.map +1 -0
- package/dist/server/mcp/classifier/BouncerClassifier.js +4 -0
- package/dist/server/mcp/classifier/BouncerClassifier.js.map +1 -0
- package/dist/server/mcp/classifier/ClaudeBouncerClassifier.d.ts +17 -0
- package/dist/server/mcp/classifier/ClaudeBouncerClassifier.d.ts.map +1 -0
- package/dist/server/mcp/classifier/ClaudeBouncerClassifier.js +142 -0
- package/dist/server/mcp/classifier/ClaudeBouncerClassifier.js.map +1 -0
- package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.d.ts +68 -0
- package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.d.ts.map +1 -0
- package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.js +182 -0
- package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.js.map +1 -0
- package/dist/server/mcp/classifier/factory.d.ts +70 -0
- package/dist/server/mcp/classifier/factory.d.ts.map +1 -0
- package/dist/server/mcp/classifier/factory.js +155 -0
- package/dist/server/mcp/classifier/factory.js.map +1 -0
- package/dist/server/mcp/server.js +52 -0
- package/dist/server/mcp/server.js.map +1 -1
- package/dist/server/routes/index.d.ts +1 -0
- package/dist/server/routes/index.d.ts.map +1 -1
- package/dist/server/routes/index.js +1 -0
- package/dist/server/routes/index.js.map +1 -1
- package/dist/server/routes/internal.d.ts +16 -0
- package/dist/server/routes/internal.d.ts.map +1 -0
- package/dist/server/routes/internal.js +94 -0
- package/dist/server/routes/internal.js.map +1 -0
- package/dist/server/services/plan/agent-resolver.d.ts +26 -0
- package/dist/server/services/plan/agent-resolver.d.ts.map +1 -0
- package/dist/server/services/plan/agent-resolver.js +102 -0
- package/dist/server/services/plan/agent-resolver.js.map +1 -0
- package/dist/server/services/plan/composer.d.ts.map +1 -1
- package/dist/server/services/plan/composer.js +59 -11
- package/dist/server/services/plan/composer.js.map +1 -1
- package/dist/server/services/plan/executor.d.ts.map +1 -1
- package/dist/server/services/plan/executor.js +3 -1
- package/dist/server/services/plan/executor.js.map +1 -1
- package/dist/server/services/plan/issue-prompt-builder.d.ts.map +1 -1
- package/dist/server/services/plan/issue-prompt-builder.js +33 -1
- package/dist/server/services/plan/issue-prompt-builder.js.map +1 -1
- package/dist/server/services/plan/parser-core.d.ts.map +1 -1
- package/dist/server/services/plan/parser-core.js +1 -0
- package/dist/server/services/plan/parser-core.js.map +1 -1
- package/dist/server/services/plan/types.d.ts +1 -0
- package/dist/server/services/plan/types.d.ts.map +1 -1
- package/dist/server/services/runtime-info.d.ts +3 -0
- package/dist/server/services/runtime-info.d.ts.map +1 -0
- package/dist/server/services/runtime-info.js +21 -0
- package/dist/server/services/runtime-info.js.map +1 -0
- package/dist/server/services/settings.d.ts +76 -2
- package/dist/server/services/settings.d.ts.map +1 -1
- package/dist/server/services/settings.js +127 -4
- package/dist/server/services/settings.js.map +1 -1
- package/dist/server/services/websocket/ask-user-question-bridge.d.ts +32 -0
- package/dist/server/services/websocket/ask-user-question-bridge.d.ts.map +1 -0
- package/dist/server/services/websocket/ask-user-question-bridge.js +115 -0
- package/dist/server/services/websocket/ask-user-question-bridge.js.map +1 -0
- package/dist/server/services/websocket/git-branch-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/git-branch-handlers.js +19 -6
- package/dist/server/services/websocket/git-branch-handlers.js.map +1 -1
- package/dist/server/services/websocket/handler.d.ts +25 -1
- package/dist/server/services/websocket/handler.d.ts.map +1 -1
- package/dist/server/services/websocket/handler.js +84 -2
- package/dist/server/services/websocket/handler.js.map +1 -1
- package/dist/server/services/websocket/quality-complexity.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-complexity.js +78 -26
- package/dist/server/services/websocket/quality-complexity.js.map +1 -1
- package/dist/server/services/websocket/quality-eta.d.ts +47 -0
- package/dist/server/services/websocket/quality-eta.d.ts.map +1 -0
- package/dist/server/services/websocket/quality-eta.js +110 -0
- package/dist/server/services/websocket/quality-eta.js.map +1 -0
- package/dist/server/services/websocket/quality-grading.d.ts +27 -4
- package/dist/server/services/websocket/quality-grading.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-grading.js +369 -201
- package/dist/server/services/websocket/quality-grading.js.map +1 -1
- package/dist/server/services/websocket/quality-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-handlers.js +145 -7
- package/dist/server/services/websocket/quality-handlers.js.map +1 -1
- package/dist/server/services/websocket/quality-operations.d.ts +34 -0
- package/dist/server/services/websocket/quality-operations.d.ts.map +1 -0
- package/dist/server/services/websocket/quality-operations.js +47 -0
- package/dist/server/services/websocket/quality-operations.js.map +1 -0
- package/dist/server/services/websocket/quality-persistence.d.ts +9 -0
- package/dist/server/services/websocket/quality-persistence.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-persistence.js +10 -0
- package/dist/server/services/websocket/quality-persistence.js.map +1 -1
- package/dist/server/services/websocket/quality-review-agent.d.ts +1 -1
- package/dist/server/services/websocket/quality-review-agent.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-review-agent.js +105 -56
- package/dist/server/services/websocket/quality-review-agent.js.map +1 -1
- package/dist/server/services/websocket/quality-service.d.ts +9 -1
- package/dist/server/services/websocket/quality-service.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-service.js +334 -14
- package/dist/server/services/websocket/quality-service.js.map +1 -1
- package/dist/server/services/websocket/quality-tools.d.ts +21 -0
- package/dist/server/services/websocket/quality-tools.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-tools.js +49 -0
- package/dist/server/services/websocket/quality-tools.js.map +1 -1
- package/dist/server/services/websocket/quality-types.d.ts +35 -2
- package/dist/server/services/websocket/quality-types.d.ts.map +1 -1
- package/dist/server/services/websocket/quality-types.js +1 -1
- package/dist/server/services/websocket/quality-types.js.map +1 -1
- package/dist/server/services/websocket/session-handlers.d.ts +3 -1
- package/dist/server/services/websocket/session-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/session-handlers.js +60 -9
- package/dist/server/services/websocket/session-handlers.js.map +1 -1
- package/dist/server/services/websocket/session-history.js +3 -0
- package/dist/server/services/websocket/session-history.js.map +1 -1
- package/dist/server/services/websocket/session-initialization.d.ts.map +1 -1
- package/dist/server/services/websocket/session-initialization.js +158 -42
- package/dist/server/services/websocket/session-initialization.js.map +1 -1
- package/dist/server/services/websocket/session-registry.d.ts +25 -0
- package/dist/server/services/websocket/session-registry.d.ts.map +1 -1
- package/dist/server/services/websocket/session-registry.js +19 -0
- package/dist/server/services/websocket/session-registry.js.map +1 -1
- package/dist/server/services/websocket/settings-handlers.d.ts +1 -1
- package/dist/server/services/websocket/settings-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/settings-handlers.js +35 -4
- package/dist/server/services/websocket/settings-handlers.js.map +1 -1
- package/dist/server/services/websocket/tab-broadcast.d.ts +7 -2
- package/dist/server/services/websocket/tab-broadcast.d.ts.map +1 -1
- package/dist/server/services/websocket/tab-broadcast.js +10 -2
- package/dist/server/services/websocket/tab-broadcast.js.map +1 -1
- package/dist/server/services/websocket/tab-event-buffer.d.ts +97 -8
- package/dist/server/services/websocket/tab-event-buffer.d.ts.map +1 -1
- package/dist/server/services/websocket/tab-event-buffer.js +138 -12
- package/dist/server/services/websocket/tab-event-buffer.js.map +1 -1
- package/dist/server/services/websocket/tab-event-replay.d.ts +29 -13
- package/dist/server/services/websocket/tab-event-replay.d.ts.map +1 -1
- package/dist/server/services/websocket/tab-event-replay.js +55 -2
- package/dist/server/services/websocket/tab-event-replay.js.map +1 -1
- package/dist/server/services/websocket/tab-handlers.d.ts +9 -1
- package/dist/server/services/websocket/tab-handlers.d.ts.map +1 -1
- package/dist/server/services/websocket/tab-handlers.js +47 -2
- package/dist/server/services/websocket/tab-handlers.js.map +1 -1
- package/dist/server/services/websocket/types.d.ts +67 -7
- package/dist/server/services/websocket/types.d.ts.map +1 -1
- package/dist/server/services/websocket/types.js +12 -6
- package/dist/server/services/websocket/types.js.map +1 -1
- package/package.json +5 -3
- package/server/cli/eta-estimator.ts +249 -0
- package/server/cli/headless/claude-invoker-process.ts +9 -1
- package/server/cli/headless/mcp-config.ts +30 -5
- package/server/cli/headless/runner.ts +21 -0
- package/server/cli/headless/stall-assessor.ts +93 -0
- package/server/cli/headless/tool-watchdog.ts +21 -0
- package/server/cli/headless/types.ts +16 -1
- package/server/cli/improvisation-history-store.ts +4 -1
- package/server/cli/improvisation-output-queue.ts +29 -7
- package/server/cli/improvisation-session-manager.ts +63 -1
- package/server/cli/improvisation-types.ts +9 -0
- package/server/cli/retry/retry-runner-factory.ts +1 -0
- package/server/engines/EngineEvent.ts +156 -0
- package/server/engines/claude/ClaudeCodeEngine.ts +404 -0
- package/server/engines/factory.ts +176 -0
- package/server/engines/opencode/OpenCodeEngine.ts +786 -0
- package/server/engines/opencode/OpenCodeServerManager.ts +577 -0
- package/server/engines/opencode/model-catalog.ts +217 -0
- package/server/engines/types.ts +173 -0
- package/server/index.ts +9 -1
- package/server/mcp/bouncer-haiku.ts +21 -145
- package/server/mcp/bouncer-integration.ts +107 -5
- package/server/mcp/classifier/BouncerClassifier.ts +40 -0
- package/server/mcp/classifier/ClaudeBouncerClassifier.ts +189 -0
- package/server/mcp/classifier/OpenCodeBouncerClassifier.ts +305 -0
- package/server/mcp/classifier/factory.ts +195 -0
- package/server/mcp/server.ts +57 -0
- package/server/routes/index.ts +1 -0
- package/server/routes/internal.ts +112 -0
- package/server/services/plan/agent-resolver.ts +115 -0
- package/server/services/plan/agents/code-review.md +38 -8
- package/server/services/plan/composer.ts +63 -11
- package/server/services/plan/executor.ts +3 -1
- package/server/services/plan/issue-prompt-builder.ts +39 -1
- package/server/services/plan/parser-core.ts +1 -0
- package/server/services/plan/types.ts +4 -0
- package/server/services/runtime-info.ts +24 -0
- package/server/services/settings.ts +161 -4
- package/server/services/websocket/ask-user-question-bridge.ts +148 -0
- package/server/services/websocket/git-branch-handlers.ts +20 -6
- package/server/services/websocket/handler.ts +89 -2
- package/server/services/websocket/quality-complexity.ts +80 -26
- package/server/services/websocket/quality-eta.ts +155 -0
- package/server/services/websocket/quality-grading.ts +445 -222
- package/server/services/websocket/quality-handlers.ts +153 -7
- package/server/services/websocket/quality-operations.ts +72 -0
- package/server/services/websocket/quality-persistence.ts +17 -0
- package/server/services/websocket/quality-review-agent.ts +154 -64
- package/server/services/websocket/quality-service.ts +361 -13
- package/server/services/websocket/quality-tools.ts +51 -0
- package/server/services/websocket/quality-types.ts +41 -2
- package/server/services/websocket/session-handlers.ts +67 -10
- package/server/services/websocket/session-history.ts +3 -0
- package/server/services/websocket/session-initialization.ts +189 -46
- package/server/services/websocket/session-registry.ts +37 -0
- package/server/services/websocket/settings-handlers.ts +41 -4
- package/server/services/websocket/tab-broadcast.ts +10 -2
- package/server/services/websocket/tab-event-buffer.ts +143 -11
- package/server/services/websocket/tab-event-replay.ts +70 -3
- package/server/services/websocket/tab-handlers.ts +53 -5
- package/server/services/websocket/types.ts +85 -7
|
@@ -0,0 +1,404 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
// Licensed under the MIT License. See LICENSE file for details.
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* ClaudeCodeEngine
|
|
6
|
+
*
|
|
7
|
+
* Thin adapter that wraps the existing Claude Code subprocess path
|
|
8
|
+
* (claude-invoker) behind the CodingAgentEngine interface. All stdout
|
|
9
|
+
* parsing, watchdog, stall detection, and process spawning stays in the
|
|
10
|
+
* headless runner modules — this file only translates HeadlessRunner
|
|
11
|
+
* callbacks into EngineEvents and owns its own process-group map for
|
|
12
|
+
* cancellation.
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import type { ChildProcess } from 'node:child_process';
|
|
16
|
+
import { executeClaudeCommand } from '../../cli/headless/claude-invoker.js';
|
|
17
|
+
import { killProcessGroup } from '../../cli/headless/runner.js';
|
|
18
|
+
import type {
|
|
19
|
+
ExecutionResult,
|
|
20
|
+
ImageAttachment,
|
|
21
|
+
ResolvedHeadlessConfig,
|
|
22
|
+
ToolUseEvent,
|
|
23
|
+
} from '../../cli/headless/types.js';
|
|
24
|
+
import type {
|
|
25
|
+
EngineEvent,
|
|
26
|
+
EngineId,
|
|
27
|
+
} from '../EngineEvent.js';
|
|
28
|
+
import type {
|
|
29
|
+
CodingAgentEngine,
|
|
30
|
+
EngineUsage,
|
|
31
|
+
PromptAttachment,
|
|
32
|
+
StartSessionOptions,
|
|
33
|
+
} from '../types.js';
|
|
34
|
+
|
|
35
|
+
type Resolver = (r: IteratorResult<EngineEvent>) => void;
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Implementation of CodingAgentEngine backed by the Claude Code headless
|
|
39
|
+
* runner. The engine holds a single logical conversation — subsequent
|
|
40
|
+
* prompts automatically continue the Claude session via --resume.
|
|
41
|
+
*/
|
|
42
|
+
export class ClaudeCodeEngine implements CodingAgentEngine {
|
|
43
|
+
readonly engineId: EngineId = 'claude-code';
|
|
44
|
+
|
|
45
|
+
private sessionOptions: StartSessionOptions | null = null;
|
|
46
|
+
private claudeSessionId: string | undefined;
|
|
47
|
+
/**
|
|
48
|
+
* Live child processes spawned by the current prompt. Keyed by pid; values
|
|
49
|
+
* are the raw ChildProcess handles. executeClaudeCommand registers the
|
|
50
|
+
* spawned process here and removes it on 'close'. cancel() iterates this
|
|
51
|
+
* map and SIGTERMs each process group — matching the pre-engine behavior
|
|
52
|
+
* from HeadlessRunner.cleanup().
|
|
53
|
+
*/
|
|
54
|
+
private readonly runningProcesses: Map<number, ChildProcess> = new Map();
|
|
55
|
+
private currentRunPromise: Promise<ExecutionResult> | null = null;
|
|
56
|
+
|
|
57
|
+
private disposed = false;
|
|
58
|
+
private iteratorDone = false;
|
|
59
|
+
private readonly queue: EngineEvent[] = [];
|
|
60
|
+
private readonly pending: Resolver[] = [];
|
|
61
|
+
|
|
62
|
+
private usage: EngineUsage = {
|
|
63
|
+
inputTokens: 0,
|
|
64
|
+
outputTokens: 0,
|
|
65
|
+
lastUpdatedAt: Date.now(),
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
async startSession(options: StartSessionOptions): Promise<void> {
|
|
69
|
+
if (this.sessionOptions) {
|
|
70
|
+
throw new Error('ClaudeCodeEngine: startSession called more than once');
|
|
71
|
+
}
|
|
72
|
+
if (this.disposed) {
|
|
73
|
+
throw new Error('ClaudeCodeEngine: cannot start a disposed engine');
|
|
74
|
+
}
|
|
75
|
+
this.sessionOptions = options;
|
|
76
|
+
this.claudeSessionId = options.resumeSessionId;
|
|
77
|
+
this.usage = {
|
|
78
|
+
inputTokens: 0,
|
|
79
|
+
outputTokens: 0,
|
|
80
|
+
lastUpdatedAt: Date.now(),
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
async sendPrompt(prompt: string, attachments?: PromptAttachment[]): Promise<void> {
|
|
85
|
+
if (!this.sessionOptions) {
|
|
86
|
+
throw new Error('ClaudeCodeEngine: sendPrompt called before startSession');
|
|
87
|
+
}
|
|
88
|
+
if (this.disposed) {
|
|
89
|
+
throw new Error('ClaudeCodeEngine: sendPrompt called after dispose');
|
|
90
|
+
}
|
|
91
|
+
if (this.currentRunPromise) {
|
|
92
|
+
throw new Error('ClaudeCodeEngine: another prompt is already in flight');
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
const opts = this.sessionOptions;
|
|
96
|
+
const imageAttachments = convertImageAttachments(attachments);
|
|
97
|
+
|
|
98
|
+
const toolStartTimes = new Map<string, number>();
|
|
99
|
+
const toolNames = new Map<string, string>();
|
|
100
|
+
const toolInputs = new Map<string, Record<string, unknown>>();
|
|
101
|
+
|
|
102
|
+
const config: ResolvedHeadlessConfig = {
|
|
103
|
+
workingDir: opts.workingDir,
|
|
104
|
+
tokenBudgetThreshold: 170_000,
|
|
105
|
+
maxSessions: 50,
|
|
106
|
+
maxRetries: 3,
|
|
107
|
+
claudeCommand: process.env.CLAUDE_COMMAND || 'claude',
|
|
108
|
+
verbose: false,
|
|
109
|
+
noColor: false,
|
|
110
|
+
improvisationMode: false,
|
|
111
|
+
movementNumber: 0,
|
|
112
|
+
directPrompt: prompt,
|
|
113
|
+
promptContext: { accumulatedKnowledge: '', filesModified: [] },
|
|
114
|
+
stallWarningMs: 300_000,
|
|
115
|
+
stallKillMs: 1_800_000,
|
|
116
|
+
stallAssessEnabled: true,
|
|
117
|
+
stallMaxExtensions: 3,
|
|
118
|
+
stallHardCapMs: 3_600_000,
|
|
119
|
+
enableToolWatchdog: true,
|
|
120
|
+
maxAutoRetries: 2,
|
|
121
|
+
model: opts.model,
|
|
122
|
+
effortLevel: opts.effortLevel,
|
|
123
|
+
continueSession: !!this.claudeSessionId,
|
|
124
|
+
claudeSessionId: this.claudeSessionId,
|
|
125
|
+
imageAttachments,
|
|
126
|
+
disallowedTools: opts.disallowedTools,
|
|
127
|
+
deployMode: opts.deployMode,
|
|
128
|
+
extraEnv: opts.extraEnv,
|
|
129
|
+
outputCallback: (text: string) => {
|
|
130
|
+
this.emit({
|
|
131
|
+
kind: 'message.delta',
|
|
132
|
+
sessionId: this.sessionIdForEvent(),
|
|
133
|
+
timestamp: Date.now(),
|
|
134
|
+
text,
|
|
135
|
+
});
|
|
136
|
+
},
|
|
137
|
+
thinkingCallback: (text: string) => {
|
|
138
|
+
this.emit({
|
|
139
|
+
kind: 'message.thinking',
|
|
140
|
+
sessionId: this.sessionIdForEvent(),
|
|
141
|
+
timestamp: Date.now(),
|
|
142
|
+
text,
|
|
143
|
+
});
|
|
144
|
+
},
|
|
145
|
+
toolUseCallback: (event: ToolUseEvent) => {
|
|
146
|
+
this.handleToolUseEvent(event, toolStartTimes, toolNames, toolInputs);
|
|
147
|
+
},
|
|
148
|
+
tokenUsageCallback: (usage: { inputTokens: number; outputTokens: number }) => {
|
|
149
|
+
this.usage = {
|
|
150
|
+
inputTokens: usage.inputTokens,
|
|
151
|
+
outputTokens: usage.outputTokens,
|
|
152
|
+
lastUpdatedAt: Date.now(),
|
|
153
|
+
};
|
|
154
|
+
this.emit({
|
|
155
|
+
kind: 'usage.update',
|
|
156
|
+
sessionId: this.sessionIdForEvent(),
|
|
157
|
+
timestamp: Date.now(),
|
|
158
|
+
inputTokens: usage.inputTokens,
|
|
159
|
+
outputTokens: usage.outputTokens,
|
|
160
|
+
});
|
|
161
|
+
},
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
const runPromise = executeClaudeCommand(prompt, 'engine', 1, {
|
|
165
|
+
config,
|
|
166
|
+
runningProcesses: this.runningProcesses,
|
|
167
|
+
});
|
|
168
|
+
this.currentRunPromise = runPromise;
|
|
169
|
+
|
|
170
|
+
runPromise.then(
|
|
171
|
+
(result) => this.onRunCompleted(result),
|
|
172
|
+
(err) => this.onRunFailed(err),
|
|
173
|
+
);
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
async cancel(): Promise<void> {
|
|
177
|
+
if (this.runningProcesses.size === 0 && !this.currentRunPromise) return;
|
|
178
|
+
this.killAllRunningProcesses();
|
|
179
|
+
if (this.currentRunPromise) {
|
|
180
|
+
await this.currentRunPromise.catch(() => { /* surfaced as engine.error */ });
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
getUsage(): EngineUsage {
|
|
185
|
+
return { ...this.usage };
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
async dispose(): Promise<void> {
|
|
189
|
+
if (this.disposed) return;
|
|
190
|
+
this.disposed = true;
|
|
191
|
+
this.killAllRunningProcesses();
|
|
192
|
+
if (this.currentRunPromise) {
|
|
193
|
+
await this.currentRunPromise.catch(() => { /* surfaced as engine.error */ });
|
|
194
|
+
}
|
|
195
|
+
this.closeIterator();
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
[Symbol.asyncIterator](): AsyncIterator<EngineEvent> {
|
|
199
|
+
return {
|
|
200
|
+
next: (): Promise<IteratorResult<EngineEvent>> => {
|
|
201
|
+
if (this.queue.length > 0) {
|
|
202
|
+
return Promise.resolve({ value: this.queue.shift() as EngineEvent, done: false });
|
|
203
|
+
}
|
|
204
|
+
if (this.iteratorDone) {
|
|
205
|
+
return Promise.resolve({ value: undefined, done: true });
|
|
206
|
+
}
|
|
207
|
+
return new Promise<IteratorResult<EngineEvent>>((resolve) => {
|
|
208
|
+
this.pending.push(resolve);
|
|
209
|
+
});
|
|
210
|
+
},
|
|
211
|
+
return: (): Promise<IteratorResult<EngineEvent>> => {
|
|
212
|
+
this.closeIterator();
|
|
213
|
+
return Promise.resolve({ value: undefined, done: true });
|
|
214
|
+
},
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
// ---------- private ----------
|
|
219
|
+
|
|
220
|
+
private sessionIdForEvent(): string {
|
|
221
|
+
return this.claudeSessionId ?? 'pending';
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
/**
|
|
225
|
+
* SIGTERM every tracked child process group; schedule a SIGKILL fallback
|
|
226
|
+
* for any that haven't exited in 5s. Mirrors HeadlessRunner.cleanup().
|
|
227
|
+
*/
|
|
228
|
+
private killAllRunningProcesses(): void {
|
|
229
|
+
if (this.runningProcesses.size === 0) return;
|
|
230
|
+
const pids = new Set<number>();
|
|
231
|
+
for (const pid of this.runningProcesses.keys()) {
|
|
232
|
+
pids.add(pid);
|
|
233
|
+
killProcessGroup(pid, 'SIGTERM');
|
|
234
|
+
}
|
|
235
|
+
setTimeout(() => {
|
|
236
|
+
for (const [pid, proc] of this.runningProcesses) {
|
|
237
|
+
if (pids.has(pid) && !proc.killed) {
|
|
238
|
+
killProcessGroup(pid, 'SIGKILL');
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
}, 5000).unref?.();
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
private emit(event: EngineEvent): void {
|
|
245
|
+
if (this.iteratorDone) return;
|
|
246
|
+
const resolver = this.pending.shift();
|
|
247
|
+
if (resolver) {
|
|
248
|
+
resolver({ value: event, done: false });
|
|
249
|
+
} else {
|
|
250
|
+
this.queue.push(event);
|
|
251
|
+
}
|
|
252
|
+
if (event.kind === 'engine.error' && event.fatal) {
|
|
253
|
+
this.closeIterator();
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
private closeIterator(): void {
|
|
258
|
+
if (this.iteratorDone) return;
|
|
259
|
+
this.iteratorDone = true;
|
|
260
|
+
const waiting = this.pending.splice(0);
|
|
261
|
+
for (const resolve of waiting) {
|
|
262
|
+
resolve({ value: undefined, done: true });
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
private onRunCompleted(result: ExecutionResult): void {
|
|
267
|
+
this.currentRunPromise = null;
|
|
268
|
+
if (result.claudeSessionId) {
|
|
269
|
+
this.claudeSessionId = result.claudeSessionId;
|
|
270
|
+
}
|
|
271
|
+
// Fall back to the definitive end-of-run usage — the streaming callback
|
|
272
|
+
// may have missed the final numbers if the result event arrived after
|
|
273
|
+
// the last message_delta.
|
|
274
|
+
if (result.apiTokenUsage) {
|
|
275
|
+
const u = result.apiTokenUsage;
|
|
276
|
+
if (u.inputTokens > this.usage.inputTokens || u.outputTokens > this.usage.outputTokens) {
|
|
277
|
+
this.usage = {
|
|
278
|
+
inputTokens: Math.max(u.inputTokens, this.usage.inputTokens),
|
|
279
|
+
outputTokens: Math.max(u.outputTokens, this.usage.outputTokens),
|
|
280
|
+
lastUpdatedAt: Date.now(),
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
this.emit({
|
|
285
|
+
kind: 'session.idle',
|
|
286
|
+
sessionId: this.sessionIdForEvent(),
|
|
287
|
+
timestamp: Date.now(),
|
|
288
|
+
stopReason: result.stopReason,
|
|
289
|
+
});
|
|
290
|
+
if (result.exitCode !== 0 && !result.assistantResponse && !result.toolUseHistory?.length) {
|
|
291
|
+
const message = result.error || `Claude exited with code ${result.exitCode}`;
|
|
292
|
+
this.emit({
|
|
293
|
+
kind: 'engine.error',
|
|
294
|
+
sessionId: this.sessionIdForEvent(),
|
|
295
|
+
timestamp: Date.now(),
|
|
296
|
+
code: 'CLAUDE_RUN_ERROR',
|
|
297
|
+
message,
|
|
298
|
+
fatal: false,
|
|
299
|
+
});
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
private onRunFailed(err: unknown): void {
|
|
304
|
+
this.currentRunPromise = null;
|
|
305
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
306
|
+
this.emit({
|
|
307
|
+
kind: 'engine.error',
|
|
308
|
+
sessionId: this.sessionIdForEvent(),
|
|
309
|
+
timestamp: Date.now(),
|
|
310
|
+
code: 'CLAUDE_SPAWN_ERROR',
|
|
311
|
+
message,
|
|
312
|
+
fatal: true,
|
|
313
|
+
});
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
private handleToolUseEvent(
|
|
317
|
+
event: ToolUseEvent,
|
|
318
|
+
starts: Map<string, number>,
|
|
319
|
+
names: Map<string, string>,
|
|
320
|
+
inputs: Map<string, Record<string, unknown>>,
|
|
321
|
+
): void {
|
|
322
|
+
if (!event.toolId) return;
|
|
323
|
+
if (event.type === 'tool_start') {
|
|
324
|
+
this.onToolStart(event, starts, names);
|
|
325
|
+
} else if (event.type === 'tool_complete') {
|
|
326
|
+
this.onToolComplete(event, names, inputs);
|
|
327
|
+
} else if (event.type === 'tool_result') {
|
|
328
|
+
this.onToolResult(event, starts, names, inputs);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
private onToolStart(
|
|
333
|
+
event: ToolUseEvent,
|
|
334
|
+
starts: Map<string, number>,
|
|
335
|
+
names: Map<string, string>,
|
|
336
|
+
): void {
|
|
337
|
+
const ts = Date.now();
|
|
338
|
+
const toolId = event.toolId as string;
|
|
339
|
+
starts.set(toolId, ts);
|
|
340
|
+
if (event.toolName) names.set(toolId, event.toolName);
|
|
341
|
+
this.emit({
|
|
342
|
+
kind: 'tool.start',
|
|
343
|
+
sessionId: this.sessionIdForEvent(),
|
|
344
|
+
timestamp: ts,
|
|
345
|
+
toolCallId: toolId,
|
|
346
|
+
toolName: event.toolName ?? '',
|
|
347
|
+
input: {},
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
private onToolComplete(
|
|
352
|
+
event: ToolUseEvent,
|
|
353
|
+
names: Map<string, string>,
|
|
354
|
+
inputs: Map<string, Record<string, unknown>>,
|
|
355
|
+
): void {
|
|
356
|
+
const toolId = event.toolId as string;
|
|
357
|
+
if (event.completeInput) inputs.set(toolId, event.completeInput);
|
|
358
|
+
if (event.toolName) names.set(toolId, event.toolName);
|
|
359
|
+
// tool.end is emitted on tool_result so the result payload is included.
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
private onToolResult(
|
|
363
|
+
event: ToolUseEvent,
|
|
364
|
+
starts: Map<string, number>,
|
|
365
|
+
names: Map<string, string>,
|
|
366
|
+
inputs: Map<string, Record<string, unknown>>,
|
|
367
|
+
): void {
|
|
368
|
+
const ts = Date.now();
|
|
369
|
+
const toolId = event.toolId as string;
|
|
370
|
+
const start = starts.get(toolId) ?? ts;
|
|
371
|
+
this.emit({
|
|
372
|
+
kind: 'tool.end',
|
|
373
|
+
sessionId: this.sessionIdForEvent(),
|
|
374
|
+
timestamp: ts,
|
|
375
|
+
toolCallId: toolId,
|
|
376
|
+
toolName: names.get(toolId) ?? '',
|
|
377
|
+
input: inputs.get(toolId) ?? {},
|
|
378
|
+
result: event.result ?? '',
|
|
379
|
+
isError: event.isError ?? false,
|
|
380
|
+
durationMs: ts - start,
|
|
381
|
+
});
|
|
382
|
+
starts.delete(toolId);
|
|
383
|
+
names.delete(toolId);
|
|
384
|
+
inputs.delete(toolId);
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
function convertImageAttachments(
|
|
389
|
+
attachments?: PromptAttachment[],
|
|
390
|
+
): ImageAttachment[] | undefined {
|
|
391
|
+
if (!attachments || attachments.length === 0) return undefined;
|
|
392
|
+
const images: ImageAttachment[] = [];
|
|
393
|
+
for (const a of attachments) {
|
|
394
|
+
if (!a.isImage || !a.base64Content) continue;
|
|
395
|
+
images.push({
|
|
396
|
+
fileName: a.fileName,
|
|
397
|
+
filePath: a.filePath ?? a.fileName,
|
|
398
|
+
content: a.base64Content,
|
|
399
|
+
isImage: true,
|
|
400
|
+
mimeType: a.mimeType,
|
|
401
|
+
});
|
|
402
|
+
}
|
|
403
|
+
return images.length > 0 ? images : undefined;
|
|
404
|
+
}
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
// Copyright (c) 2025-present Mstro, Inc. All rights reserved.
|
|
2
|
+
// Licensed under the MIT License. See LICENSE file for details.
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Engine factory — returns a concrete CodingAgentEngine for the requested
|
|
6
|
+
* EngineId. Callers pass the value of `settings.engine` and the factory
|
|
7
|
+
* dispatches to the matching implementation:
|
|
8
|
+
*
|
|
9
|
+
* - 'claude-code' → ClaudeCodeEngine (headless runner, stdout JSON)
|
|
10
|
+
* - 'opencode' → OpenCodeEngine (OpenCode SDK + SSE)
|
|
11
|
+
*
|
|
12
|
+
* The OpenCode path is backed by a single process-lifetime
|
|
13
|
+
* OpenCodeServerManager that owns the `opencode serve` subprocess. The
|
|
14
|
+
* factory itself stays synchronous; manager startup is awaited inside
|
|
15
|
+
* `startSession` via `LazyOpenCodeEngine` (below) so callers observe the
|
|
16
|
+
* same lifecycle as ClaudeCodeEngine.
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import { isEngineSwapEnabled } from '../services/settings.js';
|
|
20
|
+
import { ClaudeCodeEngine } from './claude/ClaudeCodeEngine.js';
|
|
21
|
+
import { OpenCodeEngine } from './opencode/OpenCodeEngine.js';
|
|
22
|
+
import { OpenCodeServerManager } from './opencode/OpenCodeServerManager.js';
|
|
23
|
+
import type {
|
|
24
|
+
CodingAgentEngine,
|
|
25
|
+
EngineEvent,
|
|
26
|
+
EngineId,
|
|
27
|
+
EngineUsage,
|
|
28
|
+
PromptAttachment,
|
|
29
|
+
StartSessionOptions,
|
|
30
|
+
} from './types.js';
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Process-lifetime singleton for the `opencode serve` subprocess. Created
|
|
34
|
+
* lazily on the first request for an opencode engine so Claude-only
|
|
35
|
+
* deployments never spawn the binary. `registerProcessHandlers` is set so
|
|
36
|
+
* the subprocess exits with the CLI — no orphan processes on SIGINT.
|
|
37
|
+
*/
|
|
38
|
+
let sharedOpenCodeManager: OpenCodeServerManager | null = null;
|
|
39
|
+
|
|
40
|
+
function getSharedOpenCodeServerManager(): OpenCodeServerManager {
|
|
41
|
+
if (!sharedOpenCodeManager) {
|
|
42
|
+
sharedOpenCodeManager = new OpenCodeServerManager({
|
|
43
|
+
registerProcessHandlers: true,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
return sharedOpenCodeManager;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Reset the cached OpenCode manager. Primarily for tests — never called
|
|
51
|
+
* by production code. Does not shut down the previous manager; callers
|
|
52
|
+
* that need a clean state should `shutdown()` first.
|
|
53
|
+
*/
|
|
54
|
+
export function __resetSharedOpenCodeServerManagerForTests(): void {
|
|
55
|
+
sharedOpenCodeManager = null;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Thin adapter that defers `OpenCodeEngine` construction until
|
|
60
|
+
* `startSession` runs. The real engine requires an already-bound
|
|
61
|
+
* `OpencodeClient`, but the underlying HTTP server is spawned
|
|
62
|
+
* asynchronously by `OpenCodeServerManager`. `startSession` is the first
|
|
63
|
+
* async call on the engine lifecycle, so we await `manager.start()`
|
|
64
|
+
* there, then construct the inner engine once the client is available
|
|
65
|
+
* and forward every subsequent call to it.
|
|
66
|
+
*
|
|
67
|
+
* The wrapper preserves the public `CodingAgentEngine` contract:
|
|
68
|
+
* - `engineId` is stable at `'opencode'` from construction.
|
|
69
|
+
* - Methods called before `startSession` resolve reject with the same
|
|
70
|
+
* error wording the inner engine would have produced.
|
|
71
|
+
* - `dispose()` is idempotent and tolerates the uninitialized case.
|
|
72
|
+
*/
|
|
73
|
+
class LazyOpenCodeEngine implements CodingAgentEngine {
|
|
74
|
+
readonly engineId: EngineId = 'opencode';
|
|
75
|
+
|
|
76
|
+
private inner: OpenCodeEngine | null = null;
|
|
77
|
+
private started = false;
|
|
78
|
+
private disposed = false;
|
|
79
|
+
|
|
80
|
+
constructor(private readonly manager: OpenCodeServerManager) {}
|
|
81
|
+
|
|
82
|
+
async startSession(options: StartSessionOptions): Promise<void> {
|
|
83
|
+
if (this.disposed) {
|
|
84
|
+
throw new Error('OpenCodeEngine: cannot start a disposed engine');
|
|
85
|
+
}
|
|
86
|
+
if (this.started) {
|
|
87
|
+
throw new Error('OpenCodeEngine: startSession called more than once');
|
|
88
|
+
}
|
|
89
|
+
await this.manager.start();
|
|
90
|
+
const client = this.manager.getClient();
|
|
91
|
+
this.inner = new OpenCodeEngine({
|
|
92
|
+
client,
|
|
93
|
+
directory: options.workingDir,
|
|
94
|
+
});
|
|
95
|
+
await this.inner.startSession(options);
|
|
96
|
+
this.started = true;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
sendPrompt(
|
|
100
|
+
prompt: string,
|
|
101
|
+
attachments?: PromptAttachment[],
|
|
102
|
+
): Promise<void> {
|
|
103
|
+
if (this.disposed) {
|
|
104
|
+
return Promise.reject(
|
|
105
|
+
new Error('OpenCodeEngine: sendPrompt called after dispose'),
|
|
106
|
+
);
|
|
107
|
+
}
|
|
108
|
+
if (!this.inner) {
|
|
109
|
+
return Promise.reject(
|
|
110
|
+
new Error('OpenCodeEngine: sendPrompt called before startSession'),
|
|
111
|
+
);
|
|
112
|
+
}
|
|
113
|
+
return this.inner.sendPrompt(prompt, attachments);
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
cancel(): Promise<void> {
|
|
117
|
+
if (!this.inner) return Promise.resolve();
|
|
118
|
+
return this.inner.cancel();
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
getUsage(): EngineUsage {
|
|
122
|
+
if (!this.inner) {
|
|
123
|
+
return { inputTokens: 0, outputTokens: 0, lastUpdatedAt: Date.now() };
|
|
124
|
+
}
|
|
125
|
+
return this.inner.getUsage();
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
async dispose(): Promise<void> {
|
|
129
|
+
if (this.disposed) return;
|
|
130
|
+
this.disposed = true;
|
|
131
|
+
if (this.inner) {
|
|
132
|
+
await this.inner.dispose();
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
[Symbol.asyncIterator](): AsyncIterator<EngineEvent> {
|
|
137
|
+
// The contract only allows iteration after `startSession` has resolved,
|
|
138
|
+
// so `this.inner` is guaranteed to be set when consumers begin the
|
|
139
|
+
// for-await loop. We delegate directly to the inner engine's iterator
|
|
140
|
+
// to preserve its ordering and terminal-event semantics.
|
|
141
|
+
if (!this.inner) {
|
|
142
|
+
return {
|
|
143
|
+
next: () => Promise.resolve({ value: undefined, done: true }),
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
return this.inner[Symbol.asyncIterator]();
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* Construct a new engine instance for the given engine id. The returned
|
|
152
|
+
* engine is uninitialized — the caller must call `startSession` before
|
|
153
|
+
* any other method.
|
|
154
|
+
*
|
|
155
|
+
* Feature-flag gate: when `engineSwap` is disabled, this returns
|
|
156
|
+
* `ClaudeCodeEngine` for every id. That guarantees the pre-OpenCode
|
|
157
|
+
* behavior — in particular, `LazyOpenCodeEngine` is never constructed, so
|
|
158
|
+
* the shared `OpenCodeServerManager` is never touched and no `opencode
|
|
159
|
+
* serve` subprocess is spawned. The flag is checked on every call (rather
|
|
160
|
+
* than cached) so runtime toggles take effect on the next session start.
|
|
161
|
+
*/
|
|
162
|
+
export function createEngine(engineId: EngineId): CodingAgentEngine {
|
|
163
|
+
if (!isEngineSwapEnabled()) {
|
|
164
|
+
return new ClaudeCodeEngine();
|
|
165
|
+
}
|
|
166
|
+
switch (engineId) {
|
|
167
|
+
case 'claude-code':
|
|
168
|
+
return new ClaudeCodeEngine();
|
|
169
|
+
case 'opencode':
|
|
170
|
+
return new LazyOpenCodeEngine(getSharedOpenCodeServerManager());
|
|
171
|
+
default: {
|
|
172
|
+
const exhaustive: never = engineId;
|
|
173
|
+
throw new Error(`Unknown engine id: ${String(exhaustive)}`);
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
}
|