Compare commits
3 Commits
v2.6.8-age
...
v2.6.9-war
| Author | SHA1 | Date | |
|---|---|---|---|
| f619ae0978 | |||
| 0d3d08f5f2 | |||
| 0658d19b64 |
@@ -2,6 +2,10 @@
|
|||||||
|
|
||||||
All notable changes per release tag. Most recent on top, ordered by tag creation date (which matches the git history). Tag names follow `vMAJOR.MINOR.PATCH-slug` — the slug describes what shipped, so the tag name alone is enough to recall the batch.
|
All notable changes per release tag. Most recent on top, ordered by tag creation date (which matches the git history). Tag names follow `vMAJOR.MINOR.PATCH-slug` — the slug describes what shipped, so the tag name alone is enough to recall the batch.
|
||||||
|
|
||||||
|
## v2.6.9-warm-acp — 2026-05-31
|
||||||
|
|
||||||
|
v2.6 Phase 2: goose and qwen now run as **warm ACP backends** instead of one-shot-per-task. A new `WarmAcpBackend` (`backends/warm-acp.ts`, implementing the same `AgentBackend` interface as the opencode warm server) holds one persistent `goose acp` / `qwen --acp` child + `ClientSideConnection` + ACP session per `(chat, agent)`, running `initialize` + `session/new` once and reusing the connection across turns; per-turn abort cancels the in-flight prompt (`session/cancel`) without killing the child, and a child exit marks `agent_sessions.status='crashed'` for re-spawn on the next turn. The dispatcher routes `goose`/`qwen` chat-tab tasks to the pooled warm backend via a pure `shouldUseWarmBackend(task)` predicate (warm only when both `session_id` and `chat_id` are set), keeping the one-shot `runExternalAgent` path as the fallback for session-less creators (arena, MCP, `new_task`); broker frames + `persistExternalAgentTurn` + the latest-wins `pending_changes` diff are identical to the opencode path. The `acp-dispatch.ts` `handleSessionUpdate` switch was extracted into a pure shared `acp-event-map.ts` mapper used by both the one-shot and warm paths (one-shot behavior byte-identical, all existing acp tests green). The design's `unstable_resumeSession` concern is resolved — the installed `@agentclientprotocol/sdk@^0.22.1` exposes stable `resumeSession`/`loadSession`, but resume is moot in the hot path (warm reuse needs none); cross-restart resume + idle eviction are deferred to Phase 3. Built test-first (15 new tests: `warm-acp-routing`, `acp-event-map`); 180 coder tests pass, tsc + build clean. **Smoke 2/2b (live two-message warm reuse + the opencode→boocode→opencode switch round-trip) to be run post-deploy.** Phase 3 (lifecycle hardening) is the last v2.6 phase.
|
||||||
|
|
||||||
## v2.6.8-agent-attribution — 2026-05-31
|
## v2.6.8-agent-attribution — 2026-05-31
|
||||||
|
|
||||||
v2.6 Phase 1-UX: agent attribution + switch affordances over the already-shipped `pending_changes.agent` column and `agent_sessions` table (read+display, no new backend capability). **Backend:** `pending_changes.agent` is now stamped at every queue site (native write tools → `'boocode'`, dispatched external agents → the task's agent, manual RightRail create → `NULL`) and flows through `listPending`; a new `GET /api/sessions/:id/agent-sessions` route returns `[{agent,status,has_session,last_active_at}]` per `(chat,agent)` for the session's chats; and the opencode warm-server backend consumes opencode's `session.next.step.ended` events, accumulating `input_tokens`/`output_tokens`/`cost` onto the `agent_sessions` row (new columns, idempotent). **Frontend:** the BooCoder DiffPanel renders a per-row agent badge (provider icon + label; `null` → "manual") with a "Changes from X, Y" note when a pending set spans multiple agents, and the AgentComposerBar shows a resumed / history / new-session chip beside the Provider picker — gated on an optional `sessionId` prop so BooChat is unaffected — driven by a new `useAgentSessions` hook that refetches on message-complete; `providerIcon` was extracted to a shared `components/coder/providerIcons.tsx`. Built by three parallel subagents over disjoint file sets; web + coder typecheck clean, 165 coder tests pass (9 new across `opencode-usage` and `agent-sessions.routes`). U.6's persisted token totals are conversation-cumulative and not yet surfaced in the UI (deferred). Implements the U.1–U.6 "remaining" plan from the v2.6 openspec reconciliation; Phase 2 (warm ACP goose/qwen) + Phase 3 (lifecycle hardening) remain.
|
v2.6 Phase 1-UX: agent attribution + switch affordances over the already-shipped `pending_changes.agent` column and `agent_sessions` table (read+display, no new backend capability). **Backend:** `pending_changes.agent` is now stamped at every queue site (native write tools → `'boocode'`, dispatched external agents → the task's agent, manual RightRail create → `NULL`) and flows through `listPending`; a new `GET /api/sessions/:id/agent-sessions` route returns `[{agent,status,has_session,last_active_at}]` per `(chat,agent)` for the session's chats; and the opencode warm-server backend consumes opencode's `session.next.step.ended` events, accumulating `input_tokens`/`output_tokens`/`cost` onto the `agent_sessions` row (new columns, idempotent). **Frontend:** the BooCoder DiffPanel renders a per-row agent badge (provider icon + label; `null` → "manual") with a "Changes from X, Y" note when a pending set spans multiple agents, and the AgentComposerBar shows a resumed / history / new-session chip beside the Provider picker — gated on an optional `sessionId` prop so BooChat is unaffected — driven by a new `useAgentSessions` hook that refetches on message-complete; `providerIcon` was extracted to a shared `components/coder/providerIcons.tsx`. Built by three parallel subagents over disjoint file sets; web + coder typecheck clean, 165 coder tests pass (9 new across `opencode-usage` and `agent-sessions.routes`). U.6's persisted token totals are conversation-cumulative and not yet surfaced in the UI (deferred). Implements the U.1–U.6 "remaining" plan from the v2.6 openspec reconciliation; Phase 2 (warm ACP goose/qwen) + Phase 3 (lifecycle hardening) remain.
|
||||||
|
|||||||
110
apps/coder/src/services/__tests__/acp-event-map.test.ts
Normal file
110
apps/coder/src/services/__tests__/acp-event-map.test.ts
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
import { describe, it, expect } from 'vitest';
|
||||||
|
import type { SessionNotification } from '@agentclientprotocol/sdk';
|
||||||
|
import { mapSessionUpdate } from '../acp-event-map.js';
|
||||||
|
import type { AcpToolSnapshot } from '../acp-tool-snapshot.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pure event-mapping shared by the one-shot ACP dispatch (AcpStreamContext) and
|
||||||
|
* the warm ACP backend (Phase 2). Mirrors the original handleSessionUpdate switch
|
||||||
|
* verbatim but returns normalized AgentEvents instead of publishing broker frames.
|
||||||
|
*/
|
||||||
|
describe('mapSessionUpdate (shared ACP event mapping)', () => {
|
||||||
|
function note(update: SessionNotification['update']): SessionNotification {
|
||||||
|
return { sessionId: 's1', update };
|
||||||
|
}
|
||||||
|
|
||||||
|
it('maps an agent_message_chunk text → a text event', () => {
|
||||||
|
const events = mapSessionUpdate(
|
||||||
|
note({ sessionUpdate: 'agent_message_chunk', content: { type: 'text', text: 'hello' } }),
|
||||||
|
);
|
||||||
|
expect(events).toEqual([{ type: 'text', text: 'hello' }]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps an agent_thought_chunk text → a reasoning event', () => {
|
||||||
|
const events = mapSessionUpdate(
|
||||||
|
note({ sessionUpdate: 'agent_thought_chunk', content: { type: 'text', text: 'thinking' } }),
|
||||||
|
);
|
||||||
|
expect(events).toEqual([{ type: 'reasoning', text: 'thinking' }]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores non-text content on message/thought chunks', () => {
|
||||||
|
const img = mapSessionUpdate(
|
||||||
|
note({
|
||||||
|
sessionUpdate: 'agent_message_chunk',
|
||||||
|
content: { type: 'image', data: 'x', mimeType: 'image/png' },
|
||||||
|
} as never),
|
||||||
|
);
|
||||||
|
expect(img).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps a tool_call → a tool_call event with a merged snapshot', () => {
|
||||||
|
const events = mapSessionUpdate(
|
||||||
|
note({
|
||||||
|
sessionUpdate: 'tool_call',
|
||||||
|
toolCallId: 't1',
|
||||||
|
title: 'read_file',
|
||||||
|
status: 'pending',
|
||||||
|
rawInput: { path: 'a.ts' },
|
||||||
|
} as never),
|
||||||
|
);
|
||||||
|
expect(events).toHaveLength(1);
|
||||||
|
expect(events[0]!.type).toBe('tool_call');
|
||||||
|
const snap = (events[0] as { type: 'tool_call'; toolCall: AcpToolSnapshot }).toolCall;
|
||||||
|
expect(snap.toolCallId).toBe('t1');
|
||||||
|
expect(snap.title).toBe('read_file');
|
||||||
|
expect(snap.status).toBe('pending');
|
||||||
|
expect(snap.rawInput).toEqual({ path: 'a.ts' });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps a tool_call_update → a tool_update event merged over the prior snapshot', () => {
|
||||||
|
const prior = new Map<string, AcpToolSnapshot>([
|
||||||
|
['t1', { toolCallId: 't1', title: 'read_file', status: 'pending', rawInput: { path: 'a.ts' } }],
|
||||||
|
]);
|
||||||
|
const events = mapSessionUpdate(
|
||||||
|
note({
|
||||||
|
sessionUpdate: 'tool_call_update',
|
||||||
|
toolCallId: 't1',
|
||||||
|
status: 'completed',
|
||||||
|
rawOutput: 'file body',
|
||||||
|
} as never),
|
||||||
|
prior,
|
||||||
|
);
|
||||||
|
expect(events).toHaveLength(1);
|
||||||
|
expect(events[0]!.type).toBe('tool_update');
|
||||||
|
const snap = (events[0] as { type: 'tool_update'; toolCall: AcpToolSnapshot }).toolCall;
|
||||||
|
expect(snap.toolCallId).toBe('t1');
|
||||||
|
// merged: title carried from prior, status updated, output added, input retained
|
||||||
|
expect(snap.title).toBe('read_file');
|
||||||
|
expect(snap.status).toBe('completed');
|
||||||
|
expect(snap.rawOutput).toBe('file body');
|
||||||
|
expect(snap.rawInput).toEqual({ path: 'a.ts' });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('maps available_commands_update → a commands event', () => {
|
||||||
|
const events = mapSessionUpdate(
|
||||||
|
note({
|
||||||
|
sessionUpdate: 'available_commands_update',
|
||||||
|
availableCommands: [
|
||||||
|
{ name: 'plan', description: 'make a plan' },
|
||||||
|
{ name: 'review', description: null },
|
||||||
|
],
|
||||||
|
} as never),
|
||||||
|
);
|
||||||
|
expect(events).toEqual([
|
||||||
|
{
|
||||||
|
type: 'commands',
|
||||||
|
commands: [
|
||||||
|
{ name: 'plan', description: 'make a plan' },
|
||||||
|
{ name: 'review', description: undefined },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns [] for unhandled update kinds (plan, mode change)', () => {
|
||||||
|
expect(mapSessionUpdate(note({ sessionUpdate: 'plan', entries: [] } as never))).toEqual([]);
|
||||||
|
expect(
|
||||||
|
mapSessionUpdate(note({ sessionUpdate: 'current_mode_update', currentModeId: 'code' } as never)),
|
||||||
|
).toEqual([]);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -32,9 +32,9 @@ import { createAcpNdJsonStream } from './acp-stream.js';
|
|||||||
import { waitForPermissionResponse, waitForElicitationResponse, cancelPendingPermission } from './permission-waiter.js';
|
import { waitForPermissionResponse, waitForElicitationResponse, cancelPendingPermission } from './permission-waiter.js';
|
||||||
import { mergeTaskCommands, getTaskCommands } from './agent-commands-cache.js';
|
import { mergeTaskCommands, getTaskCommands } from './agent-commands-cache.js';
|
||||||
import { readWorktreeTextFile, writeWorktreeTextFile } from './acp-client-fs.js';
|
import { readWorktreeTextFile, writeWorktreeTextFile } from './acp-client-fs.js';
|
||||||
|
import { mapSessionUpdate } from './acp-event-map.js';
|
||||||
import {
|
import {
|
||||||
type AcpToolSnapshot,
|
type AcpToolSnapshot,
|
||||||
mergeToolSnapshot,
|
|
||||||
snapshotToWireToolCall,
|
snapshotToWireToolCall,
|
||||||
synthesizeCanceledSnapshots,
|
synthesizeCanceledSnapshots,
|
||||||
} from './acp-tool-snapshot.js';
|
} from './acp-tool-snapshot.js';
|
||||||
@@ -159,63 +159,47 @@ class AcpStreamContext {
|
|||||||
} as WsFrame);
|
} as WsFrame);
|
||||||
}
|
}
|
||||||
|
|
||||||
handleToolUpdate(toolCallId: string, update: Parameters<typeof mergeToolSnapshot>[1]): void {
|
|
||||||
const previous = this.toolSnapshots.get(toolCallId);
|
|
||||||
const snapshot = mergeToolSnapshot(toolCallId, update, previous);
|
|
||||||
this.toolSnapshots.set(toolCallId, snapshot);
|
|
||||||
this.publishToolSnapshot(snapshot);
|
|
||||||
}
|
|
||||||
|
|
||||||
async handleSessionUpdate(params: SessionNotification): Promise<void> {
|
async handleSessionUpdate(params: SessionNotification): Promise<void> {
|
||||||
const update = params.update;
|
// v2.6 Phase 2: the case-by-case mapping now lives in the shared, pure
|
||||||
switch (update.sessionUpdate) {
|
// `mapSessionUpdate` (reused by the warm ACP backend). This method keeps the
|
||||||
case 'agent_message_chunk': {
|
// identical broker-publishing side effects — it just translates the normalized
|
||||||
const content = update.content;
|
// AgentEvents back into the same frames it always emitted. `this.toolSnapshots`
|
||||||
if (content.type === 'text' && 'text' in content) {
|
// is the merge accumulator, so a later tool_call_update merges over its
|
||||||
const text = (content as { text: string }).text;
|
// tool_call (the prior `handleToolUpdate` behavior, byte-for-byte).
|
||||||
this.textChunks.push(text);
|
for (const event of mapSessionUpdate(params, this.toolSnapshots)) {
|
||||||
|
switch (event.type) {
|
||||||
|
case 'text':
|
||||||
|
this.textChunks.push(event.text);
|
||||||
if (this.canStream()) {
|
if (this.canStream()) {
|
||||||
this.opts.broker!.publishFrame(this.opts.sessionId!, {
|
this.opts.broker!.publishFrame(this.opts.sessionId!, {
|
||||||
type: 'delta',
|
type: 'delta',
|
||||||
message_id: this.opts.messageId!,
|
message_id: this.opts.messageId!,
|
||||||
chat_id: this.opts.chatId!,
|
chat_id: this.opts.chatId!,
|
||||||
content: text,
|
content: event.text,
|
||||||
} as WsFrame);
|
} as WsFrame);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
case 'reasoning':
|
||||||
case 'agent_thought_chunk': {
|
this.reasoningChunks.push(event.text);
|
||||||
const content = update.content;
|
|
||||||
if (content.type === 'text' && 'text' in content) {
|
|
||||||
const text = (content as { text: string }).text;
|
|
||||||
this.reasoningChunks.push(text);
|
|
||||||
if (this.canStream()) {
|
if (this.canStream()) {
|
||||||
this.opts.broker!.publishFrame(this.opts.sessionId!, {
|
this.opts.broker!.publishFrame(this.opts.sessionId!, {
|
||||||
type: 'reasoning_delta',
|
type: 'reasoning_delta',
|
||||||
message_id: this.opts.messageId!,
|
message_id: this.opts.messageId!,
|
||||||
chat_id: this.opts.chatId!,
|
chat_id: this.opts.chatId!,
|
||||||
content: text,
|
content: event.text,
|
||||||
} as WsFrame);
|
} as WsFrame);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
case 'tool_call':
|
case 'tool_call':
|
||||||
this.handleToolUpdate(update.toolCallId, update);
|
case 'tool_update':
|
||||||
|
// mapSessionUpdate already stored the merged snapshot in this.toolSnapshots.
|
||||||
|
this.publishToolSnapshot(event.toolCall);
|
||||||
break;
|
break;
|
||||||
case 'tool_call_update':
|
case 'commands':
|
||||||
this.handleToolUpdate(update.toolCallId, update);
|
if (this.opts.taskId && event.commands.length > 0) {
|
||||||
break;
|
mergeTaskCommands(this.opts.taskId, event.commands);
|
||||||
case 'available_commands_update': {
|
|
||||||
const commands = update.availableCommands.map((cmd) => ({
|
|
||||||
name: cmd.name,
|
|
||||||
description: cmd.description ?? undefined,
|
|
||||||
}));
|
|
||||||
if (this.opts.taskId && commands.length > 0) {
|
|
||||||
mergeTaskCommands(this.opts.taskId, commands);
|
|
||||||
if (this.canStream() && this.opts.sessionId) {
|
if (this.canStream() && this.opts.sessionId) {
|
||||||
const all = getTaskCommands(this.opts.taskId) ?? commands;
|
const all = getTaskCommands(this.opts.taskId) ?? event.commands;
|
||||||
this.opts.broker!.publishFrame(this.opts.sessionId, {
|
this.opts.broker!.publishFrame(this.opts.sessionId, {
|
||||||
type: 'agent_commands',
|
type: 'agent_commands',
|
||||||
task_id: this.opts.taskId,
|
task_id: this.opts.taskId,
|
||||||
@@ -226,8 +210,6 @@ class AcpStreamContext {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
68
apps/coder/src/services/acp-event-map.ts
Normal file
68
apps/coder/src/services/acp-event-map.ts
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/**
|
||||||
|
* Shared ACP session-update → normalized AgentEvent mapping.
|
||||||
|
*
|
||||||
|
* Extracted verbatim (v2.6 Phase 2) from `AcpStreamContext.handleSessionUpdate`
|
||||||
|
* in `acp-dispatch.ts` so the warm ACP backend (`backends/warm-acp.ts`) and the
|
||||||
|
* one-shot dispatch share ONE mapping. The one-shot path translates the returned
|
||||||
|
* events into broker frames itself (preserving its prior behavior byte-for-byte);
|
||||||
|
* the warm backend forwards them to the dispatcher's `ctx.onEvent` exactly like
|
||||||
|
* the opencode-server backend does. No I/O, no broker — pure, so it's unit-testable.
|
||||||
|
*
|
||||||
|
* Spec: openspec/changes/v2-6-persistent-agent-sessions/design.md §2b.
|
||||||
|
*/
|
||||||
|
import type { SessionNotification } from '@agentclientprotocol/sdk';
|
||||||
|
import type { AgentEvent } from './agent-backend.js';
|
||||||
|
import { type AcpToolSnapshot, mergeToolSnapshot } from './acp-tool-snapshot.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map one ACP `session/update` notification to zero-or-more normalized AgentEvents.
|
||||||
|
*
|
||||||
|
* `priorSnapshots` is the caller-owned tool-call snapshot accumulator (toolCallId →
|
||||||
|
* snapshot). For `tool_call` / `tool_call_update` the merged snapshot is written
|
||||||
|
* back into it (mutated in place, mirroring `AcpStreamContext.handleToolUpdate`)
|
||||||
|
* so a later `tool_call_update` merges over the earlier `tool_call`. Pass an empty
|
||||||
|
* Map for a stateless single call.
|
||||||
|
*
|
||||||
|
* Returns an array (never throws) so the caller can splat it onto `onEvent`.
|
||||||
|
*/
|
||||||
|
export function mapSessionUpdate(
|
||||||
|
params: SessionNotification,
|
||||||
|
priorSnapshots: Map<string, AcpToolSnapshot> = new Map(),
|
||||||
|
): AgentEvent[] {
|
||||||
|
const update = params.update;
|
||||||
|
switch (update.sessionUpdate) {
|
||||||
|
case 'agent_message_chunk': {
|
||||||
|
const content = update.content;
|
||||||
|
if (content.type === 'text' && 'text' in content) {
|
||||||
|
return [{ type: 'text', text: (content as { text: string }).text }];
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
case 'agent_thought_chunk': {
|
||||||
|
const content = update.content;
|
||||||
|
if (content.type === 'text' && 'text' in content) {
|
||||||
|
return [{ type: 'reasoning', text: (content as { text: string }).text }];
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
case 'tool_call': {
|
||||||
|
const snapshot = mergeToolSnapshot(update.toolCallId, update, priorSnapshots.get(update.toolCallId));
|
||||||
|
priorSnapshots.set(update.toolCallId, snapshot);
|
||||||
|
return [{ type: 'tool_call', toolCall: snapshot }];
|
||||||
|
}
|
||||||
|
case 'tool_call_update': {
|
||||||
|
const snapshot = mergeToolSnapshot(update.toolCallId, update, priorSnapshots.get(update.toolCallId));
|
||||||
|
priorSnapshots.set(update.toolCallId, snapshot);
|
||||||
|
return [{ type: 'tool_update', toolCall: snapshot }];
|
||||||
|
}
|
||||||
|
case 'available_commands_update': {
|
||||||
|
const commands = update.availableCommands.map((cmd) => ({
|
||||||
|
name: cmd.name,
|
||||||
|
description: cmd.description ?? undefined,
|
||||||
|
}));
|
||||||
|
return [{ type: 'commands', commands }];
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -70,6 +70,12 @@ export interface PromptCtx {
|
|||||||
model: string;
|
model: string;
|
||||||
signal: AbortSignal;
|
signal: AbortSignal;
|
||||||
onEvent: (e: AgentEvent) => void;
|
onEvent: (e: AgentEvent) => void;
|
||||||
|
/** Phase 2: per-turn task id, so a warm ACP backend can route permission /
|
||||||
|
* elicitation prompts back to the UI via the permission-waiter. Optional —
|
||||||
|
* the opencode-server backend (autonomous) ignores it. */
|
||||||
|
taskId?: string;
|
||||||
|
/** Phase 2: per-turn mode id (gates autonomous mode in the permission-waiter). */
|
||||||
|
modeId?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Result of a completed turn (§2). Diff/persist happen outside the backend. */
|
/** Result of a completed turn (§2). Diff/persist happen outside the backend. */
|
||||||
|
|||||||
@@ -0,0 +1,59 @@
|
|||||||
|
import { describe, it, expect } from 'vitest';
|
||||||
|
import { shouldUseWarmBackend, isTurnOkForStopReason } from '../warm-acp-routing.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Phase 2 routing predicate: which goose/qwen tasks go to the warm pool backend
|
||||||
|
* vs the existing one-shot ACP path.
|
||||||
|
*
|
||||||
|
* The warm backend is keyed (chat_id, agent) — the persistent context unit (same
|
||||||
|
* as opencode-server). A task only routes warm when it carries BOTH a session_id
|
||||||
|
* and a chat_id, i.e. it originates from a real chat tab (the coder message route
|
||||||
|
* stamps both). Session-less creators (arena, MCP-created, generic /api/tasks,
|
||||||
|
* new_task) lack chat_id/session_id and keep the one-shot worktree-per-task path,
|
||||||
|
* which never spawns a warm process.
|
||||||
|
*/
|
||||||
|
describe('shouldUseWarmBackend (Phase 2 routing)', () => {
|
||||||
|
it('routes a chat-tab task (session_id + chat_id) to the warm backend', () => {
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'qwen', session_id: 's1', chat_id: 'c1' })).toBe(true);
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'goose', session_id: 's1', chat_id: 'c1' })).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps a session-less arena/MCP task on the one-shot path', () => {
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'qwen', session_id: null, chat_id: null })).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps a task with a session but no chat on the one-shot path', () => {
|
||||||
|
// chat_id is the warm-key half; without it ensureSession would get a degenerate
|
||||||
|
// (null, agent) key, so fall back to one-shot rather than synthesize a chat.
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'goose', session_id: 's1', chat_id: null })).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps a task with a chat but no session on the one-shot path', () => {
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'qwen', session_id: null, chat_id: 'c1' })).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('only applies to warm-capable agents (goose, qwen); others never warm here', () => {
|
||||||
|
// opencode has its own dedicated warm path; native/claude/etc. are not ACP-warm.
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'opencode', session_id: 's1', chat_id: 'c1' })).toBe(false);
|
||||||
|
expect(shouldUseWarmBackend({ agent: 'claude', session_id: 's1', chat_id: 'c1' })).toBe(false);
|
||||||
|
expect(shouldUseWarmBackend({ agent: null, session_id: 's1', chat_id: 'c1' })).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('isTurnOkForStopReason (ACP stop-reason → ok/fail)', () => {
|
||||||
|
it('treats normal completions as ok', () => {
|
||||||
|
expect(isTurnOkForStopReason('end_turn')).toBe(true);
|
||||||
|
expect(isTurnOkForStopReason('max_tokens')).toBe(true);
|
||||||
|
expect(isTurnOkForStopReason('max_turn_requests')).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('treats refusal and cancelled as failures', () => {
|
||||||
|
expect(isTurnOkForStopReason('refusal')).toBe(false);
|
||||||
|
expect(isTurnOkForStopReason('cancelled')).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('defaults an absent stop reason to a successful end_turn', () => {
|
||||||
|
expect(isTurnOkForStopReason(undefined)).toBe(true);
|
||||||
|
expect(isTurnOkForStopReason(null)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
41
apps/coder/src/services/backends/warm-acp-routing.ts
Normal file
41
apps/coder/src/services/backends/warm-acp-routing.ts
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
/**
|
||||||
|
* v2.6 Phase 2 — warm-vs-one-shot routing predicate for goose/qwen.
|
||||||
|
*
|
||||||
|
* The warm ACP backend keys its persistent process + ACP session on (chat_id,
|
||||||
|
* agent) — exactly like the opencode-server backend. A task therefore only routes
|
||||||
|
* to the warm pool when it carries BOTH a `session_id` and a `chat_id`, i.e. it
|
||||||
|
* came from a real chat tab (the coder message route + skills route stamp both).
|
||||||
|
*
|
||||||
|
* Session-less creators — arena contestants, MCP-created tasks, generic
|
||||||
|
* `POST /api/tasks`, `new_task` — leave one or both null. Those keep the existing
|
||||||
|
* one-shot worktree-per-task ACP path (`runExternalAgent`), which spawns a fresh
|
||||||
|
* `goose acp` / `qwen --acp` per turn and never holds a warm process. Routing them
|
||||||
|
* warm would either synthesize a degenerate (null, agent) key or create a chat per
|
||||||
|
* arena contestant — neither is wanted, so they stay one-shot.
|
||||||
|
*
|
||||||
|
* Pure, so it's unit-testable; the dispatcher consumes it.
|
||||||
|
*/
|
||||||
|
const WARM_CAPABLE_AGENTS = new Set(['goose', 'qwen']);
|
||||||
|
|
||||||
|
export function shouldUseWarmBackend(task: {
|
||||||
|
agent: string | null;
|
||||||
|
session_id: string | null;
|
||||||
|
chat_id: string | null;
|
||||||
|
}): boolean {
|
||||||
|
if (!task.agent || !WARM_CAPABLE_AGENTS.has(task.agent)) return false;
|
||||||
|
return task.session_id != null && task.chat_id != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map an ACP prompt `stopReason` to the backend's ok/fail contract (TurnResult.ok).
|
||||||
|
*
|
||||||
|
* ACP's `StopReason` union includes normal completions (`end_turn`, `max_tokens`,
|
||||||
|
* `max_turn_requests`) and abnormal ones (`refusal`, `cancelled`). Only the latter
|
||||||
|
* two read as a failed turn; everything else (including an undefined/absent reason,
|
||||||
|
* which we default to `end_turn`) is a successful completion. Pure so it's testable
|
||||||
|
* independently of the warm process.
|
||||||
|
*/
|
||||||
|
export function isTurnOkForStopReason(stopReason: string | null | undefined): boolean {
|
||||||
|
const reason = stopReason ?? 'end_turn';
|
||||||
|
return reason !== 'refusal' && reason !== 'cancelled';
|
||||||
|
}
|
||||||
411
apps/coder/src/services/backends/warm-acp.ts
Normal file
411
apps/coder/src/services/backends/warm-acp.ts
Normal file
@@ -0,0 +1,411 @@
|
|||||||
|
/**
|
||||||
|
* v2.6 Phase 2 — WarmAcpBackend (goose, qwen).
|
||||||
|
*
|
||||||
|
* One persistent stdio process + ONE `ClientSideConnection` per (chat, agent),
|
||||||
|
* `initialize` + `session/new` done ONCE, reused across every turn — the warm
|
||||||
|
* analogue of the previous one-shot `acp-dispatch.ts` (which spawned/torn-down a
|
||||||
|
* fresh `goose acp` / `qwen --acp` per turn). Mirrors Paseo's `SpawnedACPProcess`.
|
||||||
|
*
|
||||||
|
* Implements the Phase 0 `AgentBackend` interface (same contract as
|
||||||
|
* `OpenCodeServerBackend`). Emits transport-agnostic `AgentEvent`s via the SHARED
|
||||||
|
* `mapSessionUpdate` (reused verbatim from the one-shot stack); the dispatcher maps
|
||||||
|
* those to WS frames + `persistExternalAgentTurn`, unchanged.
|
||||||
|
*
|
||||||
|
* Lifecycle decisions (design.md §2b / §10):
|
||||||
|
* - **Child lifetime is the pool's, not a request's.** Spawned once; never tied
|
||||||
|
* to a per-turn abort signal. Only the in-flight `prompt` gets `ctx.signal` —
|
||||||
|
* abort = ACP `session/cancel`, NOT killing the child.
|
||||||
|
* - **Per-turn abort** cancels the prompt on the warm connection so the SAME
|
||||||
|
* process serves the next turn.
|
||||||
|
* - **Crash** (child exit) marks `agent_sessions.status='crashed'` + logs; the
|
||||||
|
* next `ensureSession` re-spawns + re-`session/new` (Phase 3 hardens auto-restart).
|
||||||
|
* - **Resume across a process restart is NOT attempted in Phase 2.** goose ACP
|
||||||
|
* advertises no `loadSession`/`session.resume`; qwen does, but cross-restart
|
||||||
|
* resume is Phase 3. Within ONE live process the ACP session persists across
|
||||||
|
* turns (the whole point of "warm"); a restart re-`session/new` (memory loss
|
||||||
|
* across restart, accepted per §10). The agent's resume capabilities ARE
|
||||||
|
* probed and logged for forward-compat.
|
||||||
|
*
|
||||||
|
* Each WarmAcpBackend instance owns exactly one (chat, agent) — the dispatcher
|
||||||
|
* pools them under `agentPool.register(chatId, agent, backend)`.
|
||||||
|
*
|
||||||
|
* SDK note (@agentclientprotocol/sdk@^0.22.1, cross-checked against the design's
|
||||||
|
* `^0.14` worry): the resume method is the STABLE `resumeSession` (`session/resume`,
|
||||||
|
* gated by `agentCapabilities.sessionCapabilities.resume`), NOT the `^0.14`
|
||||||
|
* `unstable_resumeSession`. `loadSession` is gated by `agentCapabilities.loadSession`.
|
||||||
|
*/
|
||||||
|
import { spawn, type ChildProcess } from 'node:child_process';
|
||||||
|
import type { FastifyBaseLogger } from 'fastify';
|
||||||
|
import {
|
||||||
|
ClientSideConnection,
|
||||||
|
type Client,
|
||||||
|
type SessionNotification,
|
||||||
|
type RequestPermissionRequest,
|
||||||
|
type RequestPermissionResponse,
|
||||||
|
type ReadTextFileRequest,
|
||||||
|
type ReadTextFileResponse,
|
||||||
|
type WriteTextFileRequest,
|
||||||
|
type WriteTextFileResponse,
|
||||||
|
type CreateTerminalRequest,
|
||||||
|
type CreateTerminalResponse,
|
||||||
|
type CreateElicitationRequest,
|
||||||
|
type CreateElicitationResponse,
|
||||||
|
} from '@agentclientprotocol/sdk';
|
||||||
|
import type { Sql } from '../../db.js';
|
||||||
|
import { resolveLaunchSpec } from '../acp-spawn.js';
|
||||||
|
import { isTurnOkForStopReason } from './warm-acp-routing.js';
|
||||||
|
import { getResolvedRegistry, type ResolvedProviderDef } from '../provider-config-registry.js';
|
||||||
|
import { createAcpNdJsonStream } from '../acp-stream.js';
|
||||||
|
import { mapSessionUpdate } from '../acp-event-map.js';
|
||||||
|
import { readWorktreeTextFile, writeWorktreeTextFile } from '../acp-client-fs.js';
|
||||||
|
import { waitForPermissionResponse, waitForElicitationResponse, cancelPendingPermission } from '../permission-waiter.js';
|
||||||
|
import { type AcpToolSnapshot, synthesizeCanceledSnapshots } from '../acp-tool-snapshot.js';
|
||||||
|
import type {
|
||||||
|
AgentBackend,
|
||||||
|
AgentEvent,
|
||||||
|
AgentSessionHandle,
|
||||||
|
EnsureSessionOpts,
|
||||||
|
PromptCtx,
|
||||||
|
TurnResult,
|
||||||
|
} from '../agent-backend.js';
|
||||||
|
|
||||||
|
/** State for one in-flight turn (only one at a time per backend — turns serialize). */
|
||||||
|
interface TurnState {
|
||||||
|
/** Per-turn task id, for routing permission prompts back to the UI. */
|
||||||
|
taskId: string | undefined;
|
||||||
|
/** BooCode session id for permission-waiter's broker frames. */
|
||||||
|
sessionId: string;
|
||||||
|
/** Per-turn mode id (autonomous-mode gate in permission-waiter). */
|
||||||
|
modeId: string | undefined;
|
||||||
|
onEvent: (e: AgentEvent) => void;
|
||||||
|
/** Tool-call snapshot accumulator for this turn — merge across tool_call_update. */
|
||||||
|
snapshots: Map<string, AcpToolSnapshot>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface WarmAcpBackendDeps {
|
||||||
|
sql: Sql;
|
||||||
|
log: FastifyBaseLogger;
|
||||||
|
/** The (chat, agent) this backend serves — its pool identity + DB key. */
|
||||||
|
chatId: string;
|
||||||
|
agent: string;
|
||||||
|
/** Resolved binary for the agent (from available_agents.install_path), or null. */
|
||||||
|
installPath: string | null;
|
||||||
|
/** Optional override of the resolved registry def (defaults to a live lookup). */
|
||||||
|
resolved?: ResolvedProviderDef;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class WarmAcpBackend implements AgentBackend {
|
||||||
|
readonly backend = 'acp_warm' as const;
|
||||||
|
|
||||||
|
private readonly sql: Sql;
|
||||||
|
private readonly log: FastifyBaseLogger;
|
||||||
|
private readonly chatId: string;
|
||||||
|
private readonly agent: string;
|
||||||
|
private readonly installPath: string | null;
|
||||||
|
private readonly resolvedOverride: ResolvedProviderDef | undefined;
|
||||||
|
|
||||||
|
private child: ChildProcess | null = null;
|
||||||
|
private connection: ClientSideConnection | null = null;
|
||||||
|
/** The single ACP session id for this warm process; null until session/new. */
|
||||||
|
private acpSessionId: string | null = null;
|
||||||
|
private up = false;
|
||||||
|
/** Idempotent spawn guard — one warm process per backend, started lazily. */
|
||||||
|
private starting: Promise<void> | null = null;
|
||||||
|
/** Resume capabilities probed at initialize, logged for forward-compat (Phase 3). */
|
||||||
|
private supportsLoadSession = false;
|
||||||
|
private supportsResumeSession = false;
|
||||||
|
|
||||||
|
/** The current in-flight turn; the Client closures read it. Null between turns. */
|
||||||
|
private activeTurn: TurnState | null = null;
|
||||||
|
|
||||||
|
constructor(deps: WarmAcpBackendDeps) {
|
||||||
|
this.sql = deps.sql;
|
||||||
|
this.log = deps.log;
|
||||||
|
this.chatId = deps.chatId;
|
||||||
|
this.agent = deps.agent;
|
||||||
|
this.installPath = deps.installPath;
|
||||||
|
this.resolvedOverride = deps.resolved;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** §2: liveness for the health endpoint + dispatcher fallback decision. */
|
||||||
|
health(): 'up' | 'down' {
|
||||||
|
return this.up ? 'up' : 'down';
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── warm-process lifecycle (2.1 spawn + initialize + session/new ONCE) ───────
|
||||||
|
|
||||||
|
/** Lazy: spawn the warm process on first use. Idempotent — one process per backend. */
|
||||||
|
private ensureProcess(worktreePath: string): Promise<void> {
|
||||||
|
if (this.up && this.connection && this.acpSessionId) return Promise.resolve();
|
||||||
|
if (!this.starting) {
|
||||||
|
this.starting = this.startProcess(worktreePath).catch((err) => {
|
||||||
|
// Reset so a later ensureSession can retry the spawn after a failed start.
|
||||||
|
this.starting = null;
|
||||||
|
throw err;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return this.starting;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async startProcess(worktreePath: string): Promise<void> {
|
||||||
|
const resolved = this.resolvedOverride ?? getResolvedRegistry().get(this.agent);
|
||||||
|
const spec = resolved ? resolveLaunchSpec(resolved, this.installPath) : null;
|
||||||
|
if (!spec) throw new Error(`warm-acp: agent '${this.agent}' does not support ACP (no launch spec)`);
|
||||||
|
|
||||||
|
this.log.info({ agent: this.agent, chatId: this.chatId, binary: spec.binary, worktreePath }, 'warm-acp: spawning warm process');
|
||||||
|
// Child lifetime is the pool's. NOT tied to any per-turn abort signal — only
|
||||||
|
// the in-flight prompt is cancellable (via ACP session/cancel in prompt()).
|
||||||
|
const child = spawn(spec.binary, spec.args, {
|
||||||
|
cwd: worktreePath,
|
||||||
|
stdio: ['pipe', 'pipe', 'pipe'],
|
||||||
|
env: { ...process.env, ...spec.env },
|
||||||
|
});
|
||||||
|
this.child = child;
|
||||||
|
|
||||||
|
// 2.3: supervise the child; react to its exit, never let a request scope kill it.
|
||||||
|
child.on('exit', (code, signal) => {
|
||||||
|
this.up = false;
|
||||||
|
this.connection = null;
|
||||||
|
this.acpSessionId = null;
|
||||||
|
this.starting = null;
|
||||||
|
this.log.warn({ agent: this.agent, chatId: this.chatId, code, signal }, 'warm-acp: warm process exited — marking crashed (rebuild on next turn)');
|
||||||
|
void this.markCrashed();
|
||||||
|
});
|
||||||
|
// A spawn error (e.g. ENOENT) surfaces here, not as an exit.
|
||||||
|
child.on('error', (err) => {
|
||||||
|
this.up = false;
|
||||||
|
this.log.error({ agent: this.agent, chatId: this.chatId, err: errMsg(err) }, 'warm-acp: warm process error');
|
||||||
|
});
|
||||||
|
|
||||||
|
const stream = createAcpNdJsonStream(child);
|
||||||
|
const connection = new ClientSideConnection(() => this.buildClient(worktreePath), stream);
|
||||||
|
|
||||||
|
const init = await connection.initialize({
|
||||||
|
protocolVersion: 1,
|
||||||
|
clientInfo: { name: 'boocoder', version: '2.6.0' },
|
||||||
|
clientCapabilities: {},
|
||||||
|
});
|
||||||
|
const caps = init.agentCapabilities;
|
||||||
|
this.supportsLoadSession = caps?.loadSession === true;
|
||||||
|
this.supportsResumeSession = caps?.sessionCapabilities?.resume != null;
|
||||||
|
|
||||||
|
const session = await connection.newSession({ cwd: worktreePath, mcpServers: [] });
|
||||||
|
this.connection = connection;
|
||||||
|
this.acpSessionId = session.sessionId;
|
||||||
|
this.up = true;
|
||||||
|
this.log.info(
|
||||||
|
{
|
||||||
|
agent: this.agent,
|
||||||
|
chatId: this.chatId,
|
||||||
|
acpSessionId: session.sessionId,
|
||||||
|
loadSession: this.supportsLoadSession,
|
||||||
|
resumeSession: this.supportsResumeSession,
|
||||||
|
},
|
||||||
|
'warm-acp: warm session ready',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Build the ACP Client callbacks ONCE per connection. They read `this.activeTurn`
|
||||||
|
* so each turn's events/permissions route to the right place — exactly the
|
||||||
|
* opencode-server `activeTurn` pattern. Worktree-scoped FS like AcpStreamContext. */
|
||||||
|
private buildClient(worktreePath: string): Client {
|
||||||
|
return {
|
||||||
|
sessionUpdate: async (params: SessionNotification): Promise<void> => {
|
||||||
|
const turn = this.activeTurn;
|
||||||
|
if (!turn) return; // between turns — drop (no orphan settles a future turn)
|
||||||
|
for (const event of mapSessionUpdate(params, turn.snapshots)) {
|
||||||
|
turn.onEvent(event);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
requestPermission: async (params: RequestPermissionRequest): Promise<RequestPermissionResponse> => {
|
||||||
|
const turn = this.activeTurn;
|
||||||
|
if (turn?.taskId) {
|
||||||
|
// Route to the UI via the per-turn task id (same as the one-shot path).
|
||||||
|
return waitForPermissionResponse(turn.taskId, turn.sessionId, this.agent, turn.modeId, params);
|
||||||
|
}
|
||||||
|
const firstOption = params.options[0];
|
||||||
|
if (firstOption) return { outcome: { outcome: 'selected', optionId: firstOption.optionId } };
|
||||||
|
return { outcome: { outcome: 'cancelled' } };
|
||||||
|
},
|
||||||
|
readTextFile: async (params: ReadTextFileRequest): Promise<ReadTextFileResponse> => {
|
||||||
|
const content = await readWorktreeTextFile(worktreePath, params.path, params.line, params.limit);
|
||||||
|
return { content };
|
||||||
|
},
|
||||||
|
writeTextFile: async (params: WriteTextFileRequest): Promise<WriteTextFileResponse> => {
|
||||||
|
await writeWorktreeTextFile(worktreePath, params.path, params.content);
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
createTerminal: async (_params: CreateTerminalRequest): Promise<CreateTerminalResponse> => {
|
||||||
|
return { terminalId: 'noop' };
|
||||||
|
},
|
||||||
|
unstable_createElicitation: async (params: CreateElicitationRequest): Promise<CreateElicitationResponse> => {
|
||||||
|
const turn = this.activeTurn;
|
||||||
|
if (turn?.taskId) {
|
||||||
|
return waitForElicitationResponse(turn.taskId, turn.sessionId, this.agent, turn.modeId, params);
|
||||||
|
}
|
||||||
|
return { action: 'decline' };
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── ensureSession: create-or-reuse the warm session (2.1) ───────────────────
|
||||||
|
|
||||||
|
async ensureSession(sessionId: string, opts: EnsureSessionOpts): Promise<AgentSessionHandle> {
|
||||||
|
await this.ensureProcess(opts.worktreePath);
|
||||||
|
if (!this.acpSessionId) throw new Error('warm-acp: session not ready after ensureProcess');
|
||||||
|
|
||||||
|
// P1.5-b: agent_sessions keys on (chat_id, agent). The ACP session id is the
|
||||||
|
// resume handle WITHIN the live process; across a process restart it's stale,
|
||||||
|
// so ensureProcess re-`session/new` and we upsert the fresh id here.
|
||||||
|
await this.sql`
|
||||||
|
INSERT INTO agent_sessions
|
||||||
|
(chat_id, session_id, worktree_id, agent, backend, agent_session_id, server_port, status, last_active_at)
|
||||||
|
VALUES
|
||||||
|
(${opts.chatId}, ${sessionId}, ${opts.worktreeId}, ${opts.agent}, 'acp_warm', ${this.acpSessionId}, NULL, 'active', clock_timestamp())
|
||||||
|
ON CONFLICT (chat_id, agent) DO UPDATE SET
|
||||||
|
session_id = EXCLUDED.session_id,
|
||||||
|
worktree_id = EXCLUDED.worktree_id,
|
||||||
|
backend = 'acp_warm',
|
||||||
|
agent_session_id = EXCLUDED.agent_session_id,
|
||||||
|
server_port = NULL,
|
||||||
|
status = 'active',
|
||||||
|
last_active_at = clock_timestamp()
|
||||||
|
`.catch((err) => {
|
||||||
|
this.log.warn({ err: errMsg(err), chatId: opts.chatId, agent: opts.agent }, 'warm-acp: agent_sessions upsert failed (non-fatal)');
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
sessionId,
|
||||||
|
agent: opts.agent,
|
||||||
|
backend: 'acp_warm',
|
||||||
|
chatId: opts.chatId,
|
||||||
|
worktreeId: opts.worktreeId,
|
||||||
|
agentSessionId: this.acpSessionId,
|
||||||
|
serverPort: null,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── prompt: one turn on the warm connection (2.2) ───────────────────────────
|
||||||
|
|
||||||
|
async prompt(handle: AgentSessionHandle, input: string, ctx: PromptCtx): Promise<TurnResult> {
|
||||||
|
// The warm process may have crashed between ensureSession and here, or this
|
||||||
|
// backend was rebuilt — re-establish before prompting.
|
||||||
|
await this.ensureProcess(ctx.worktreePath);
|
||||||
|
const connection = this.connection;
|
||||||
|
const acpSessionId = this.acpSessionId;
|
||||||
|
if (!connection || !acpSessionId) {
|
||||||
|
return { ok: false, error: 'warm-acp: no live ACP connection' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const snapshots = new Map<string, AcpToolSnapshot>();
|
||||||
|
// taskId routes permission/elicitation prompts back to the UI. The dispatcher
|
||||||
|
// passes it (plus mode) on the per-turn PromptCtx; permission-waiter keys on it.
|
||||||
|
const turn: TurnState = {
|
||||||
|
taskId: ctx.taskId,
|
||||||
|
sessionId: handle.sessionId,
|
||||||
|
modeId: ctx.modeId,
|
||||||
|
onEvent: ctx.onEvent,
|
||||||
|
snapshots,
|
||||||
|
};
|
||||||
|
this.activeTurn = turn;
|
||||||
|
|
||||||
|
// Per-turn abort: cancel the in-flight prompt on the SAME connection — never
|
||||||
|
// kill the child (that's the pool's lifetime). On cancel we also synthesize
|
||||||
|
// 'canceled' updates for any still-running tool calls so the UI doesn't leave
|
||||||
|
// them spinning (mirrors AcpStreamContext.markAborted).
|
||||||
|
let aborted = false;
|
||||||
|
const onAbort = () => {
|
||||||
|
if (aborted) return;
|
||||||
|
aborted = true;
|
||||||
|
connection.cancel({ sessionId: acpSessionId }).catch(() => {});
|
||||||
|
if (ctx.taskId) cancelPendingPermission(ctx.taskId);
|
||||||
|
for (const snap of synthesizeCanceledSnapshots(snapshots.values())) {
|
||||||
|
snapshots.set(snap.toolCallId, snap);
|
||||||
|
ctx.onEvent({ type: 'tool_update', toolCall: snap });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (ctx.signal.aborted) {
|
||||||
|
this.activeTurn = null;
|
||||||
|
return { ok: false, error: 'aborted' };
|
||||||
|
}
|
||||||
|
ctx.signal.addEventListener('abort', onAbort, { once: true });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await connection.prompt({
|
||||||
|
sessionId: acpSessionId,
|
||||||
|
prompt: [{ type: 'text', text: input }],
|
||||||
|
});
|
||||||
|
if (aborted) return { ok: false, error: 'aborted' };
|
||||||
|
const stopReason = result.stopReason ?? 'end_turn';
|
||||||
|
return isTurnOkForStopReason(stopReason)
|
||||||
|
? { ok: true }
|
||||||
|
: { ok: false, error: `stop_reason: ${stopReason}` };
|
||||||
|
} catch (err) {
|
||||||
|
if (aborted) return { ok: false, error: 'aborted' };
|
||||||
|
return { ok: false, error: errMsg(err) };
|
||||||
|
} finally {
|
||||||
|
ctx.signal.removeEventListener('abort', onAbort);
|
||||||
|
this.activeTurn = null;
|
||||||
|
await this.sql`
|
||||||
|
UPDATE agent_sessions SET status = 'idle', last_active_at = clock_timestamp()
|
||||||
|
WHERE chat_id = ${this.chatId} AND agent = ${this.agent}
|
||||||
|
`.catch(() => {});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── teardown ────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
async closeSession(handle: AgentSessionHandle): Promise<void> {
|
||||||
|
// Gracefully close the ACP session if the agent supports it; then kill the child.
|
||||||
|
if (this.connection && this.acpSessionId) {
|
||||||
|
await this.connection.closeSession({ sessionId: this.acpSessionId }).catch(() => {});
|
||||||
|
}
|
||||||
|
await this.killChild();
|
||||||
|
await this.sql`
|
||||||
|
UPDATE agent_sessions SET status = 'closed'
|
||||||
|
WHERE chat_id = ${handle.chatId} AND agent = ${handle.agent}
|
||||||
|
`.catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
|
async dispose(): Promise<void> {
|
||||||
|
this.up = false;
|
||||||
|
this.activeTurn = null;
|
||||||
|
if (this.connection && this.acpSessionId) {
|
||||||
|
await this.connection.closeSession({ sessionId: this.acpSessionId }).catch(() => {});
|
||||||
|
}
|
||||||
|
await this.killChild();
|
||||||
|
this.connection = null;
|
||||||
|
this.acpSessionId = null;
|
||||||
|
this.starting = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async killChild(): Promise<void> {
|
||||||
|
const child = this.child;
|
||||||
|
this.child = null;
|
||||||
|
if (!child || child.killed) return;
|
||||||
|
child.kill('SIGTERM');
|
||||||
|
await new Promise<void>((resolve) => {
|
||||||
|
const t = setTimeout(() => {
|
||||||
|
if (!child.killed) child.kill('SIGKILL');
|
||||||
|
resolve();
|
||||||
|
}, 5_000);
|
||||||
|
t.unref?.();
|
||||||
|
child.once('close', () => {
|
||||||
|
clearTimeout(t);
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async markCrashed(): Promise<void> {
|
||||||
|
await this.sql`
|
||||||
|
UPDATE agent_sessions SET status = 'crashed'
|
||||||
|
WHERE chat_id = ${this.chatId} AND agent = ${this.agent}
|
||||||
|
`.catch(() => {});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function errMsg(e: unknown): string {
|
||||||
|
return e instanceof Error ? e.message : String(e);
|
||||||
|
}
|
||||||
@@ -14,6 +14,8 @@ import { persistExternalAgentTurn } from './agent-turn-persist.js';
|
|||||||
import { snapshotToWireToolCall, type AcpToolSnapshot } from './acp-tool-snapshot.js';
|
import { snapshotToWireToolCall, type AcpToolSnapshot } from './acp-tool-snapshot.js';
|
||||||
import { agentPool } from './agent-pool.js';
|
import { agentPool } from './agent-pool.js';
|
||||||
import { OpenCodeServerBackend } from './backends/opencode-server.js';
|
import { OpenCodeServerBackend } from './backends/opencode-server.js';
|
||||||
|
import { WarmAcpBackend } from './backends/warm-acp.js';
|
||||||
|
import { shouldUseWarmBackend } from './backends/warm-acp-routing.js';
|
||||||
import type { AgentBackend, AgentEvent } from './agent-backend.js';
|
import type { AgentBackend, AgentEvent } from './agent-backend.js';
|
||||||
|
|
||||||
interface InferenceRunner {
|
interface InferenceRunner {
|
||||||
@@ -121,10 +123,15 @@ export function createDispatcher(deps: Deps): { start(): void; stop(): Promise<v
|
|||||||
SELECT name, supports_acp, install_path FROM available_agents WHERE name = ${task.agent}
|
SELECT name, supports_acp, install_path FROM available_agents WHERE name = ${task.agent}
|
||||||
`;
|
`;
|
||||||
if (agentRow) {
|
if (agentRow) {
|
||||||
// v2.6 (1.7): opencode routes to the warm pool backend; every other
|
// v2.6 (1.7): opencode routes to its warm HTTP-server backend.
|
||||||
// external agent keeps the existing one-shot ACP/PTY path untouched.
|
// v2.6 Phase 2 (2.4): goose/qwen route to the warm ACP backend WHEN the
|
||||||
|
// task came from a real chat tab (session_id + chat_id) — shouldUseWarmBackend.
|
||||||
|
// Session-less creators (arena, MCP, new_task, generic /api/tasks) keep the
|
||||||
|
// existing one-shot worktree-per-task ACP/PTY path untouched.
|
||||||
if (task.agent === 'opencode') {
|
if (task.agent === 'opencode') {
|
||||||
await runOpenCodeServerTask(task, agentRow.install_path);
|
await runOpenCodeServerTask(task, agentRow.install_path);
|
||||||
|
} else if (shouldUseWarmBackend(task)) {
|
||||||
|
await runWarmAcpTask(task, agentRow.install_path);
|
||||||
} else {
|
} else {
|
||||||
await runExternalAgent(task, agentRow.supports_acp, agentRow.install_path);
|
await runExternalAgent(task, agentRow.supports_acp, agentRow.install_path);
|
||||||
}
|
}
|
||||||
@@ -788,6 +795,245 @@ export function createDispatcher(deps: Deps): { start(): void; stop(): Promise<v
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─── Path B (warm ACP): goose / qwen warm backend (v2.6 Phase 2) ─────────────
|
||||||
|
|
||||||
|
// Warm ACP backends are per (chat, agent): each owns ONE stdio process + ACP
|
||||||
|
// connection + session. Pool key = chatId; the AgentPool's secondary key is the
|
||||||
|
// agent. This mirrors agent_sessions' (chat_id, agent) PK.
|
||||||
|
function getWarmAcpBackend(chatId: string, agent: string, installPath: string | null): WarmAcpBackend {
|
||||||
|
let backend = agentPool.get(chatId, agent);
|
||||||
|
if (!backend) {
|
||||||
|
backend = new WarmAcpBackend({
|
||||||
|
sql,
|
||||||
|
log,
|
||||||
|
chatId,
|
||||||
|
agent,
|
||||||
|
installPath,
|
||||||
|
resolved: getResolvedRegistry().get(agent),
|
||||||
|
});
|
||||||
|
agentPool.register(chatId, agent, backend);
|
||||||
|
}
|
||||||
|
return backend as WarmAcpBackend;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function runWarmAcpTask(
|
||||||
|
task: {
|
||||||
|
id: string;
|
||||||
|
project_id: string;
|
||||||
|
input: string;
|
||||||
|
agent: string | null;
|
||||||
|
model: string | null;
|
||||||
|
mode_id: string | null;
|
||||||
|
thinking_option_id: string | null;
|
||||||
|
session_id: string | null;
|
||||||
|
chat_id: string | null;
|
||||||
|
},
|
||||||
|
installPath: string | null,
|
||||||
|
): Promise<void> {
|
||||||
|
const taskId = task.id;
|
||||||
|
const agent = task.agent!;
|
||||||
|
// shouldUseWarmBackend guarantees both non-null before we get here.
|
||||||
|
const sessionId = task.session_id!;
|
||||||
|
const chatId = task.chat_id!;
|
||||||
|
log.info({ taskId, agent, chatId }, 'dispatcher: starting task (path B — warm ACP)');
|
||||||
|
|
||||||
|
const [project] = await sql<{ path: string | null }[]>`
|
||||||
|
SELECT path FROM projects WHERE id = ${task.project_id}
|
||||||
|
`;
|
||||||
|
const projectPath = project?.path;
|
||||||
|
if (!projectPath) {
|
||||||
|
await sql`
|
||||||
|
UPDATE tasks
|
||||||
|
SET state = 'failed', ended_at = clock_timestamp(), output_summary = 'Project has no path — cannot create worktree'
|
||||||
|
WHERE id = ${taskId}
|
||||||
|
`;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ac = new AbortController();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await sql`
|
||||||
|
UPDATE tasks
|
||||||
|
SET state = 'running', started_at = clock_timestamp(), execution_path = 'acp'
|
||||||
|
WHERE id = ${taskId}
|
||||||
|
`;
|
||||||
|
|
||||||
|
// Persistent, session-keyed worktree (shared across turns + agents; NOT torn
|
||||||
|
// down per turn — Phase 3 reaps it). Same as the opencode-server path so a
|
||||||
|
// chat that switches opencode↔goose↔qwen shares one worktree.
|
||||||
|
const { worktreeId, worktreePath, baseCommit } = await ensureSessionWorktree(sql, projectPath, sessionId, {
|
||||||
|
signal: ac.signal,
|
||||||
|
});
|
||||||
|
log.info({ taskId, worktreePath }, 'dispatcher: session worktree ready (warm ACP)');
|
||||||
|
|
||||||
|
const [assistantMsg] = await sql<{ id: string }[]>`
|
||||||
|
INSERT INTO messages (session_id, chat_id, role, content, status, created_at)
|
||||||
|
VALUES (${sessionId}, ${chatId}, 'assistant', '', 'streaming', clock_timestamp())
|
||||||
|
RETURNING id
|
||||||
|
`;
|
||||||
|
const assistantId = assistantMsg!.id;
|
||||||
|
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'message_started',
|
||||||
|
message_id: assistantId,
|
||||||
|
chat_id: chatId,
|
||||||
|
role: 'assistant',
|
||||||
|
} as WsFrame);
|
||||||
|
|
||||||
|
const manifestCommands = getManifestCommands(agent);
|
||||||
|
if (manifestCommands.length > 0) {
|
||||||
|
setTaskCommands(taskId, manifestCommands);
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'agent_commands',
|
||||||
|
task_id: taskId,
|
||||||
|
session_id: sessionId,
|
||||||
|
commands: manifestCommands,
|
||||||
|
} as WsFrame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Accumulate the turn's stream for persistence + the final message content.
|
||||||
|
const textChunks: string[] = [];
|
||||||
|
const reasoningChunks: string[] = [];
|
||||||
|
const toolSnaps = new Map<string, AcpToolSnapshot>();
|
||||||
|
|
||||||
|
// Map transport-agnostic AgentEvents → the SAME WS frames the one-shot ACP
|
||||||
|
// path emits (identical to runOpenCodeServerTask's onEvent). No dcp stripping:
|
||||||
|
// that's an opencode-plugin artifact; goose/qwen don't emit dcp tags.
|
||||||
|
const onEvent = (e: AgentEvent): void => {
|
||||||
|
switch (e.type) {
|
||||||
|
case 'text':
|
||||||
|
textChunks.push(e.text);
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'delta',
|
||||||
|
message_id: assistantId,
|
||||||
|
chat_id: chatId,
|
||||||
|
content: e.text,
|
||||||
|
} as WsFrame);
|
||||||
|
break;
|
||||||
|
case 'reasoning':
|
||||||
|
reasoningChunks.push(e.text);
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'reasoning_delta',
|
||||||
|
message_id: assistantId,
|
||||||
|
chat_id: chatId,
|
||||||
|
content: e.text,
|
||||||
|
} as WsFrame);
|
||||||
|
break;
|
||||||
|
case 'tool_call':
|
||||||
|
case 'tool_update':
|
||||||
|
toolSnaps.set(e.toolCall.toolCallId, e.toolCall);
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'tool_call',
|
||||||
|
message_id: assistantId,
|
||||||
|
chat_id: chatId,
|
||||||
|
tool_call: snapshotToWireToolCall(e.toolCall),
|
||||||
|
} as WsFrame);
|
||||||
|
break;
|
||||||
|
case 'commands':
|
||||||
|
if (e.commands.length > 0) {
|
||||||
|
setTaskCommands(taskId, e.commands);
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'agent_commands',
|
||||||
|
task_id: taskId,
|
||||||
|
session_id: sessionId,
|
||||||
|
commands: e.commands,
|
||||||
|
} as WsFrame);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const model = task.model ?? undefined;
|
||||||
|
const backend = getWarmAcpBackend(chatId, agent, installPath);
|
||||||
|
const handle = await backend.ensureSession(sessionId, {
|
||||||
|
agent,
|
||||||
|
model: model ?? '',
|
||||||
|
chatId,
|
||||||
|
worktreePath,
|
||||||
|
worktreeId,
|
||||||
|
projectId: task.project_id,
|
||||||
|
});
|
||||||
|
const result = await backend.prompt(handle, task.input, {
|
||||||
|
worktreePath,
|
||||||
|
model: model ?? '',
|
||||||
|
signal: ac.signal,
|
||||||
|
onEvent,
|
||||||
|
taskId,
|
||||||
|
modeId: task.mode_id ?? undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
const assistantContent = textChunks.join('').slice(0, 50_000);
|
||||||
|
const reasoningText = reasoningChunks.join('').slice(0, 200_000);
|
||||||
|
const outputSummary = (result.ok ? textChunks.join('') : result.error ?? 'warm ACP turn failed').slice(0, 500);
|
||||||
|
|
||||||
|
await persistExternalAgentTurn(sql, assistantId, [...toolSnaps.values()], reasoningText);
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
UPDATE messages
|
||||||
|
SET content = ${assistantContent}, status = 'complete', finished_at = clock_timestamp()
|
||||||
|
WHERE id = ${assistantId}
|
||||||
|
`;
|
||||||
|
broker.publishFrame(sessionId, {
|
||||||
|
type: 'message_complete',
|
||||||
|
message_id: assistantId,
|
||||||
|
chat_id: chatId,
|
||||||
|
} as WsFrame);
|
||||||
|
|
||||||
|
if (stopping) {
|
||||||
|
await sql`UPDATE tasks SET state = 'cancelled', ended_at = clock_timestamp() WHERE id = ${taskId}`;
|
||||||
|
return; // worktree persists (no cleanup); backend stays warm
|
||||||
|
}
|
||||||
|
|
||||||
|
// Diff the persistent worktree against its captured baseline and SUPERSEDE
|
||||||
|
// the session's prior pending row (latest-wins) — identical to opencode.
|
||||||
|
const diff = await diffWorktree(worktreePath, projectPath, {
|
||||||
|
signal: ac.signal,
|
||||||
|
baseRef: baseCommit ?? 'HEAD',
|
||||||
|
});
|
||||||
|
if (diff) {
|
||||||
|
await sql`
|
||||||
|
DELETE FROM pending_changes WHERE session_id = ${sessionId} AND status = 'pending'
|
||||||
|
`;
|
||||||
|
await sql`
|
||||||
|
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff, agent)
|
||||||
|
VALUES (${sessionId}, ${taskId}, ${projectPath}, 'edit', ${diff}, ${agent})
|
||||||
|
`;
|
||||||
|
log.info({ taskId, diffLength: diff.length }, 'dispatcher: diff superseded prior pending change (warm ACP)');
|
||||||
|
} else {
|
||||||
|
log.info({ taskId }, 'dispatcher: no changes detected in session worktree (warm ACP)');
|
||||||
|
}
|
||||||
|
|
||||||
|
// NO worktree cleanup — persistent (Phase 3 reaps it). Backend stays warm.
|
||||||
|
|
||||||
|
const [extCostRow] = await sql<{ total: number | null }[]>`
|
||||||
|
SELECT SUM(tokens_used)::int AS total
|
||||||
|
FROM messages
|
||||||
|
WHERE session_id = ${sessionId} AND tokens_used IS NOT NULL
|
||||||
|
`;
|
||||||
|
const extCostTokens = extCostRow?.total ?? null;
|
||||||
|
|
||||||
|
const finalState = result.ok ? 'completed' : 'failed';
|
||||||
|
await sql`
|
||||||
|
UPDATE tasks
|
||||||
|
SET state = ${finalState}, ended_at = clock_timestamp(), output_summary = ${outputSummary}, cost_tokens = ${extCostTokens}
|
||||||
|
WHERE id = ${taskId}
|
||||||
|
`;
|
||||||
|
log.info({ taskId, agent, finalState }, 'dispatcher: task finished (warm ACP)');
|
||||||
|
clearTaskCommands(taskId);
|
||||||
|
} catch (err) {
|
||||||
|
const errMsg = err instanceof Error ? err.message : String(err);
|
||||||
|
log.error({ taskId, agent, err: errMsg }, 'dispatcher: warm ACP error');
|
||||||
|
await sql`
|
||||||
|
UPDATE tasks
|
||||||
|
SET state = 'failed', ended_at = clock_timestamp(), output_summary = ${errMsg.slice(0, 500)}
|
||||||
|
WHERE id = ${taskId}
|
||||||
|
`.catch(() => {});
|
||||||
|
clearTaskCommands(taskId);
|
||||||
|
// No worktree cleanup (persistent); backend stays warm for the next turn.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ─── Helpers ────────────────────────────────────────────────────────────────
|
// ─── Helpers ────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
async function waitForCompletion(assistantId: string): Promise<string> {
|
async function waitForCompletion(assistantId: string): Promise<string> {
|
||||||
|
|||||||
@@ -348,7 +348,7 @@ Per-session Docker sandbox spawned by BooCoder on first write. Only project path
|
|||||||
|
|
||||||
-----
|
-----
|
||||||
|
|
||||||
## Shipped (v2.2.2–v2.6.8 — interactive ACP, provider lifecycle, persistent agent sessions, workspace UX)
|
## Shipped (v2.2.2–v2.6.9 — interactive ACP, provider lifecycle, persistent agent sessions, workspace UX)
|
||||||
|
|
||||||
All tags `vMAJOR.MINOR.PATCH-slug`, monotonic per minor, assigned at ship time (planning slugs differ — see the numbering-discipline note below). `CHANGELOG.md` is the canonical per-tag record. **Note on numbering divergence:** the *planned-feature* "v2.3 — Provider lifecycle" actually shipped under the **v2.5.4–v2.5.13** tags; the *planned-feature* "v2.4 — BooCoder as ACP agent" remains **unshipped** even though v2.4.0/v2.4.1 *tags* shipped unrelated content (Unsloth lifts, sidecar routing). The patch-tag thread and the conceptual-milestone thread have diverged — read tags as the ship record, the `## v2.x` feature sections below as the milestone plan. The v2.3.0–v2.5.1 tags were never CHANGELOG-backfilled; summarized here from commit bodies.
|
All tags `vMAJOR.MINOR.PATCH-slug`, monotonic per minor, assigned at ship time (planning slugs differ — see the numbering-discipline note below). `CHANGELOG.md` is the canonical per-tag record. **Note on numbering divergence:** the *planned-feature* "v2.3 — Provider lifecycle" actually shipped under the **v2.5.4–v2.5.13** tags; the *planned-feature* "v2.4 — BooCoder as ACP agent" remains **unshipped** even though v2.4.0/v2.4.1 *tags* shipped unrelated content (Unsloth lifts, sidecar routing). The patch-tag thread and the conceptual-milestone thread have diverged — read tags as the ship record, the `## v2.x` feature sections below as the milestone plan. The v2.3.0–v2.5.1 tags were never CHANGELOG-backfilled; summarized here from commit bodies.
|
||||||
|
|
||||||
@@ -382,7 +382,8 @@ All tags `vMAJOR.MINOR.PATCH-slug`, monotonic per minor, assigned at ship time (
|
|||||||
- `v2.6.5-panes-tabs-composer` — **workspace UX batch (BooChat panes/tabs/composer + the persistence that backs it).** *Panes/tabs:* open a chat in a fresh pane (ChatTabBar "Open in new pane" + fork-beside-original via a new `open_chat_in_new_pane` event), per-pane `[+]` → New BooChat/BooTerm/BooCode menu, closing a chat pane relocates its tabs (in order) to the oldest chat/empty pane (reopen strips restored chatIds from every live pane first → no dup), stable session-scoped tab numbers (assigned on open, retired on close, never reused, map-keyed render), and the empty/landing pane became a real session history (open + separately-fetched archived chats). Removed the per-message "Open in pane" artifact button. *Persistence:* `sessions.workspace_panes` widened from bare `WorkspacePane[]` → a `WorkspaceState` envelope (`panes` + `tabNumbers`/`nextTabNumber` + `closedPaneStack`); PATCH validator zod-unions legacy-array-or-envelope and migrates on write; `session_workspace_updated` WS frame widened (web+server byte-identical, parity test green). *Composer:* morphing **Send → Stop → Queue** button keyed on `sending || activeTaskId` (folds in the standalone Stop pill, adds `cancelTask`); pasted chips trail the typed text so a leading slash stays first. *Tooling:* new read-only `read_tab_by_number` tool + an optional `ToolExecCtx` (`{ sql, sessionId }`) 4th arg on `ToolDef.execute`
|
- `v2.6.5-panes-tabs-composer` — **workspace UX batch (BooChat panes/tabs/composer + the persistence that backs it).** *Panes/tabs:* open a chat in a fresh pane (ChatTabBar "Open in new pane" + fork-beside-original via a new `open_chat_in_new_pane` event), per-pane `[+]` → New BooChat/BooTerm/BooCode menu, closing a chat pane relocates its tabs (in order) to the oldest chat/empty pane (reopen strips restored chatIds from every live pane first → no dup), stable session-scoped tab numbers (assigned on open, retired on close, never reused, map-keyed render), and the empty/landing pane became a real session history (open + separately-fetched archived chats). Removed the per-message "Open in pane" artifact button. *Persistence:* `sessions.workspace_panes` widened from bare `WorkspacePane[]` → a `WorkspaceState` envelope (`panes` + `tabNumbers`/`nextTabNumber` + `closedPaneStack`); PATCH validator zod-unions legacy-array-or-envelope and migrates on write; `session_workspace_updated` WS frame widened (web+server byte-identical, parity test green). *Composer:* morphing **Send → Stop → Queue** button keyed on `sending || activeTaskId` (folds in the standalone Stop pill, adds `cancelTask`); pasted chips trail the typed text so a leading slash stays first. *Tooling:* new read-only `read_tab_by_number` tool + an optional `ToolExecCtx` (`{ sql, sessionId }`) 4th arg on `ToolDef.execute`
|
||||||
- `v2.6.6-claude-md` — docs-only CLAUDE.md session-learnings from the v2.6.5 batch: the `WorkspaceState` envelope migration, the `ToolExecCtx` plumb (`read_tab_by_number` as reference), the two-schema-files-one-DB ownership split + idempotent `confdeltype` FK-action-flip pattern, and React-StrictMode nested-`setState` idempotency
|
- `v2.6.6-claude-md` — docs-only CLAUDE.md session-learnings from the v2.6.5 batch: the `WorkspaceState` envelope migration, the `ToolExecCtx` plumb (`read_tab_by_number` as reference), the two-schema-files-one-DB ownership split + idempotent `confdeltype` FK-action-flip pattern, and React-StrictMode nested-`setState` idempotency
|
||||||
- `v2.6.7-interrupt-guard` — **F.1 fix:** post-interrupt stale-terminal bug in the opencode warm-server backend (one-click reachable since `v2.6.5`'s Stop button). opencode emits one trailing `session.idle`/`session.error` for a cancelled turn (sessionID only, no turn id) that settled the *next* turn early as success. Pure per-session guard (`backends/turn-guard.ts` — arm-on-abort / swallow-one-orphan / self-heal-on-activity) wired into `opencode-server.ts`; 3 regression tests (TDD). First item of the v2.6 openspec "remaining" plan; Phase 1-UX / 2 / 3 still open
|
- `v2.6.7-interrupt-guard` — **F.1 fix:** post-interrupt stale-terminal bug in the opencode warm-server backend (one-click reachable since `v2.6.5`'s Stop button). opencode emits one trailing `session.idle`/`session.error` for a cancelled turn (sessionID only, no turn id) that settled the *next* turn early as success. Pure per-session guard (`backends/turn-guard.ts` — arm-on-abort / swallow-one-orphan / self-heal-on-activity) wired into `opencode-server.ts`; 3 regression tests (TDD). First item of the v2.6 openspec "remaining" plan; Phase 1-UX / 2 / 3 still open
|
||||||
- `v2.6.8-agent-attribution` — **v2.6 Phase 1-UX** (U.1–U.6), built by 3 parallel subagents over disjoint files. Backend: `pending_changes.agent` stamped at every queue site + flows through `listPending`; new `GET /api/sessions/:id/agent-sessions` route; opencode warm-server consumes `session.next.step.ended` → accumulates `input_tokens`/`output_tokens`/`cost` on `agent_sessions`. Frontend: DiffPanel per-row agent badges + multi-agent note; AgentComposerBar resumed/history/new-session chip (gated on optional `sessionId`, BooChat unaffected); shared `providerIcons.tsx` + `useAgentSessions` hook. 9 new tests; web+coder tsc clean. **Backend deploys via boocoder restart; frontend awaits the `boocode` Docker rebuild.** Phase 2/3 remain
|
- `v2.6.8-agent-attribution` — **v2.6 Phase 1-UX** (U.1–U.6), built by 3 parallel subagents over disjoint files. Backend: `pending_changes.agent` stamped at every queue site + flows through `listPending`; new `GET /api/sessions/:id/agent-sessions` route; opencode warm-server consumes `session.next.step.ended` → accumulates `input_tokens`/`output_tokens`/`cost` on `agent_sessions`. Frontend: DiffPanel per-row agent badges + multi-agent note; AgentComposerBar resumed/history/new-session chip (gated on optional `sessionId`, BooChat unaffected); shared `providerIcons.tsx` + `useAgentSessions` hook. 9 new tests; web+coder tsc clean. Both surfaces deployed (boocoder restart + `boocode` Docker rebuild). Phase 2/3 remain
|
||||||
|
- `v2.6.9-warm-acp` — **v2.6 Phase 2:** goose/qwen run as **warm ACP backends** (one persistent `goose acp`/`qwen --acp` child + `ClientSideConnection` + ACP session per `(chat,agent)`, `initialize`+`session/new` once, reused across turns) instead of one-shot. New `WarmAcpBackend` (same `AgentBackend` interface as opencode); abort = `session/cancel` the prompt only (never kills the child); dispatcher routes goose/qwen chat-tab tasks via pure `shouldUseWarmBackend` (one-shot fallback kept for arena/MCP/`new_task`); `handleSessionUpdate` extracted to a shared pure `acp-event-map.ts` (one-shot path byte-identical). SDK concern resolved (`@agentclientprotocol/sdk@^0.22.1` has stable resume; moot warm, deferred to Phase 3). 15 new tests, 180 coder tests pass. Backend-only deploy (boocoder restart). **Smoke 2/2b pending live.** Phase 3 (lifecycle hardening) is the last v2.6 phase
|
||||||
|
|
||||||
-----
|
-----
|
||||||
|
|
||||||
|
|||||||
@@ -40,16 +40,14 @@ ACP follows; hardening last.
|
|||||||
- [ ] **Smoke U:** stage edits with opencode then boocode → DiffPanel badges each row to the
|
- [ ] **Smoke U:** stage edits with opencode then boocode → DiffPanel badges each row to the
|
||||||
right agent; composer shows "resumed" when re-selecting opencode, "new session" for goose. *(pending live frontend deploy — Docker container rebuild)*
|
right agent; composer shows "resumed" when re-selecting opencode, "new session" for goose. *(pending live frontend deploy — Docker container rebuild)*
|
||||||
|
|
||||||
## Phase 2 — Warm ACP backend (goose, qwen) — ⬜ REMAINING
|
## Phase 2 — Warm ACP backend (goose, qwen) — ✅ SHIPPED `v2.6.9-warm-acp` (Smoke 2/2b pending live)
|
||||||
|
|
||||||
> **Lift (design §10):** `qwen --acp` is a validated reference (real stdio multi-session, `loadSession`/resume) — wire qwen into the existing `acp-dispatch.ts` stack. **goose ACP has no `loadSession`/resume** → cross-restart resume needs a different design (re-`session/new` + accept memory loss, or replay). Cross-check qwen `@agentclientprotocol/sdk@^0.14` vs BooCode `^0.22` before relying on `unstable_resumeSession`. Do **qwen first** to de-risk.
|
> **Lift (design §10):** `qwen --acp` is a validated reference (real stdio multi-session, `loadSession`/resume) — wire qwen into the existing `acp-dispatch.ts` stack. **goose ACP has no `loadSession`/resume** → cross-restart resume needs a different design (re-`session/new` + accept memory loss, or replay). Cross-check qwen `@agentclientprotocol/sdk@^0.14` vs BooCode `^0.22` before relying on `unstable_resumeSession`. Do **qwen first** to de-risk.
|
||||||
|
|
||||||
- [ ] 2.1 `backends/warm-acp.ts`: persistent spawn + `ClientSideConnection`; `initialize` +
|
- [x] 2.1 `backends/warm-acp.ts` `WarmAcpBackend` — persistent spawn + `ClientSideConnection`; `initialize` + `session/new` once per `(chat,agent)`. `handleSessionUpdate` extracted to a shared pure `acp-event-map.ts` (one-shot path byte-identical).
|
||||||
`session/new` once; reuse `acp-dispatch.ts` `handleSessionUpdate`.
|
- [x] 2.2 `prompt`: `session/prompt` on the warm connection per turn; abort = `session/cancel` the prompt only (never kills the child).
|
||||||
- [ ] 2.2 `prompt`: `session/prompt` on the warm connection per turn; per-turn abort signal only.
|
- [x] 2.3 Child supervision: pool-owned lifetime; `exit` marks `agent_sessions.status='crashed'` → re-spawn next turn.
|
||||||
- [ ] 2.3 Child supervision: detached lifetime, exit handler marks `status='crashed'`.
|
- [x] 2.4 Dispatcher routes `goose`/`qwen` chat-tab tasks to the warm backend via pure `shouldUseWarmBackend(task)` (needs `session_id`+`chat_id`); one-shot `runExternalAgent` fallback kept for arena/MCP/`new_task`. *(SDK note resolved: installed `@agentclientprotocol/sdk@^0.22.1` has stable `resumeSession`/`loadSession`; resume moot in the warm hot path, deferred to Phase 3.)*
|
||||||
- [ ] 2.4 Dispatcher routes `goose`/`qwen` to warm backend; keep one-shot fallback for arena/MCP
|
|
||||||
(or opt those into pool too — decide in review).
|
|
||||||
- [ ] **Smoke 2:** two messages in a goose chat reuse the same process + ACP session + worktree;
|
- [ ] **Smoke 2:** two messages in a goose chat reuse the same process + ACP session + worktree;
|
||||||
reasoning still renders; no per-turn respawn.
|
reasoning still renders; no per-turn respawn.
|
||||||
- [ ] **Smoke 2b (switch round-trip):** opencode → boocode → opencode in one chat — opencode
|
- [ ] **Smoke 2b (switch round-trip):** opencode → boocode → opencode in one chat — opencode
|
||||||
@@ -99,7 +97,7 @@ ACP follows; hardening last.
|
|||||||
|
|
||||||
1. ~~**F.1 interrupt-bug fix**~~ — ✅ shipped `v2.6.7-interrupt-guard` (3 regression tests, TDD).
|
1. ~~**F.1 interrupt-bug fix**~~ — ✅ shipped `v2.6.7-interrupt-guard` (3 regression tests, TDD).
|
||||||
2. ~~**Phase 1-UX** (U.1–U.6)~~ — ✅ shipped `v2.6.8-agent-attribution` (3 parallel agents, disjoint files; 9 new tests). Smoke U pending the frontend Docker rebuild.
|
2. ~~**Phase 1-UX** (U.1–U.6)~~ — ✅ shipped `v2.6.8-agent-attribution` (3 parallel agents, disjoint files; 9 new tests). Smoke U pending the frontend Docker rebuild.
|
||||||
3. **Phase 2 — warm ACP, qwen first then goose** — qwen has a validated `--acp` reference; goose's missing resume is the open design question, so qwen de-risks the pattern. Smoke 2 + 2b (the switch round-trip success criterion).
|
3. ~~**Phase 2 — warm ACP, qwen first then goose**~~ — ✅ shipped `v2.6.9-warm-acp` (15 new tests; one-shot path preserved). Smoke 2 + 2b pending live exercise post-deploy.
|
||||||
4. **Phase 3 — lifecycle hardening** — lift openchamber's state machine; do crash-recovery (3.1/3.2/3.6) + worktree reaper (3.3/3.4 + LRU) together (shared supervision loop). Closes the two ⬜ success criteria (server-crash recovery, close→cleanup).
|
4. **Phase 3 — lifecycle hardening** — lift openchamber's state machine; do crash-recovery (3.1/3.2/3.6) + worktree reaper (3.3/3.4 + LRU) together (shared supervision loop). Closes the two ⬜ success criteria (server-crash recovery, close→cleanup).
|
||||||
5. **Tests T.1–T.3 + `BOOCODER.md` (D.1 remainder)** — backfill alongside each phase, not at the end.
|
5. **Tests T.1–T.3 + `BOOCODER.md` (D.1 remainder)** — backfill alongside each phase, not at the end.
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user