feat(coder): Phase 1-UX backend — agent attribution + agent-sessions route + opencode usage
pending_changes.agent stamped at every queue site (native -> 'boocode', dispatched external -> task.agent, manual RightRail -> NULL) + flows through listPending. New GET /api/sessions/:id/agent-sessions -> [{agent,status,has_session,last_active_at}] per (chat,agent). opencode warm server consumes session.next.step.ended, accumulating input_tokens/output_tokens/cost onto agent_sessions (new idempotent columns) via a pure opencode-usage.ts mapper. Tests: agent-sessions.routes (3) + opencode-usage (6); tsc clean.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -13,6 +13,10 @@ export interface PendingChange {
|
||||
operation: 'create' | 'edit' | 'delete';
|
||||
diff: string;
|
||||
status: 'pending' | 'applied' | 'rejected' | 'reverted';
|
||||
// v2.6 Phase 1-UX: which agent staged this change (DiffPanel attribution).
|
||||
// Native boocode write tools stamp 'boocode'; the manual RightRail create path
|
||||
// passes null (renders as "manual"). NULL on legacy rows queued pre-v2.6.
|
||||
agent: string | null;
|
||||
created_at: string;
|
||||
}
|
||||
|
||||
@@ -34,13 +38,17 @@ export async function queueEdit(
|
||||
oldString: string,
|
||||
newString: string,
|
||||
projectRoot: string,
|
||||
// v2.6 Phase 1-UX: attribution. Defaults to 'boocode' because the only callers
|
||||
// that omit it are the native write tools (edit_file/create_file/delete_file).
|
||||
// Pass null explicitly for the manual RightRail create path.
|
||||
agent: string | null = 'boocode',
|
||||
): Promise<PendingChange> {
|
||||
const resolved = resolveWritePath(projectRoot, filePath);
|
||||
const diff = JSON.stringify({ old: oldString, new: newString });
|
||||
|
||||
const [row] = await sql<PendingChange[]>`
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'edit', ${diff})
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff, agent)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'edit', ${diff}, ${agent})
|
||||
RETURNING *
|
||||
`;
|
||||
return row!;
|
||||
@@ -53,12 +61,15 @@ export async function queueCreate(
|
||||
filePath: string,
|
||||
content: string,
|
||||
projectRoot: string,
|
||||
// See queueEdit: defaults to 'boocode' for the native write tools; the manual
|
||||
// RightRail create route passes null.
|
||||
agent: string | null = 'boocode',
|
||||
): Promise<PendingChange> {
|
||||
const resolved = resolveWritePath(projectRoot, filePath);
|
||||
|
||||
const [row] = await sql<PendingChange[]>`
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'create', ${content})
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff, agent)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'create', ${content}, ${agent})
|
||||
RETURNING *
|
||||
`;
|
||||
return row!;
|
||||
@@ -70,12 +81,14 @@ export async function queueDelete(
|
||||
taskId: string | null,
|
||||
filePath: string,
|
||||
projectRoot: string,
|
||||
// See queueEdit: defaults to 'boocode' for the native write tools.
|
||||
agent: string | null = 'boocode',
|
||||
): Promise<PendingChange> {
|
||||
const resolved = resolveWritePath(projectRoot, filePath);
|
||||
|
||||
const [row] = await sql<PendingChange[]>`
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'delete', '')
|
||||
INSERT INTO pending_changes (session_id, task_id, file_path, operation, diff, agent)
|
||||
VALUES (${sessionId}, ${taskId}, ${resolved}, 'delete', '', ${agent})
|
||||
RETURNING *
|
||||
`;
|
||||
return row!;
|
||||
|
||||
Reference in New Issue
Block a user