gadget/gadget-code/src/lib/drone-session.ts
2026-05-11 19:07:48 -04:00

528 lines
16 KiB
TypeScript

// src/lib/drone-session.ts
// Copyright (C) 2026 Robert Colbert <rob.colbert@openplatform.us>
// All Rights Reserved
import {
GadgetComponent,
GadgetLogLevel,
IUser,
IDroneRegistration,
ChatTurnStatus,
GadgetId,
WorkspaceMode,
IChatToolCall,
IChatTurnBlock,
} from "@gadget/api";
import {
GadgetSocket,
SocketSession,
SocketSessionType,
} from "./socket-session.js";
import { SocketService } from "../services/index.js";
import { ChatTurn } from "../models/chat-turn.js";
interface IStreamingBuffer {
currentMode: 'thinking' | 'responding' | null;
thinkingContent: string;
respondingContent: string;
lastBlockCreatedAt?: Date;
}
export class DroneSession extends SocketSession {
protected type: SocketSessionType = SocketSessionType.Drone;
registration: IDroneRegistration;
chatSessionId: GadgetId | undefined;
currentTurnId: GadgetId | undefined;
workspaceMode: WorkspaceMode = WorkspaceMode.Idle;
private streamingBuffers: Map<string, IStreamingBuffer> = new Map();
constructor(socket: GadgetSocket, registration: IDroneRegistration) {
super(socket, registration.user as IUser);
this.registration = registration;
}
register() {
super.register();
this.socket.on("status", this.onStatus.bind(this));
this.socket.on(
"workspaceModeChanged",
this.onWorkspaceModeChanged.bind(this),
);
this.socket.on("thinking", this.onThinking.bind(this));
this.socket.on("response", this.onResponse.bind(this));
this.socket.on("toolCall", this.onToolCall.bind(this));
this.socket.on("workOrderComplete", this.onWorkOrderComplete.bind(this));
this.socket.on(
"requestCrashRecovery",
this.onRequestCrashRecovery.bind(this),
);
this.socket.on("requestTermination", this.onRequestTermination.bind(this));
this.socket.on("log", this.onLog.bind(this));
this.socket.on("agent:thinking", this.onAgentThinking.bind(this));
this.socket.on("agent:response", this.onAgentResponse.bind(this));
this.socket.on("agent:tool-call", this.onAgentToolCall.bind(this));
this.socket.on("agent:tool-result", this.onAgentToolResult.bind(this));
this.socket.on("agent:complete", this.onAgentComplete.bind(this));
}
async onLog(
timestamp: Date,
component: GadgetComponent,
level: GadgetLogLevel,
message: string,
metadata?: unknown,
): Promise<void> {
if (!this.chatSessionId) {
return;
}
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onLog(timestamp, component, level, message, metadata);
} catch (error) {
this.log.error("failed to route log message", { error });
}
}
async onStatus(message: string): Promise<void> {
if (!this.chatSessionId) {
this.log.warn(
"drone status event received but no chat session is active",
);
return;
}
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.socket.emit("status", message);
} catch (error) {
this.log.error("failed to route status message", { error });
}
}
/**
* Called when the drone emits thinking content from the agent.
* Aggregates thinking tokens in memory and persists at mode changes.
*/
async onThinking(content: string): Promise<void> {
if (!this.chatSessionId) {
this.log.warn("thinking event received but no chat session is active");
return;
}
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onThinking(content);
if (this.currentTurnId) {
const buffer = this.getOrCreateBuffer(this.currentTurnId);
// Check for mode transition
if (buffer.currentMode !== 'thinking') {
// Flush previous mode if exists
await this.flushBuffer(this.currentTurnId);
buffer.currentMode = 'thinking';
buffer.thinkingContent = '';
buffer.lastBlockCreatedAt = new Date();
}
// Aggregate content
buffer.thinkingContent += content;
}
} catch (error) {
this.log.error("failed to route thinking event", { error });
}
}
/**
* Called when the drone emits response content from the agent.
* Aggregates response tokens in memory and persists at mode changes.
*/
async onResponse(content: string): Promise<void> {
if (!this.chatSessionId) {
this.log.warn("response event received but no chat session is active");
return;
}
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onResponse(content);
if (this.currentTurnId) {
const buffer = this.getOrCreateBuffer(this.currentTurnId);
// Check for mode transition
if (buffer.currentMode !== 'responding') {
// Flush previous mode if exists
await this.flushBuffer(this.currentTurnId);
buffer.currentMode = 'responding';
buffer.respondingContent = '';
buffer.lastBlockCreatedAt = new Date();
}
// Aggregate content
buffer.respondingContent += content;
}
} catch (error) {
this.log.error("failed to route response event", { error });
}
}
/**
* Called when the drone emits a tool call event from the agent.
* Flushes current buffer and adds tool block immediately.
*/
async onToolCall(
callId: string,
name: string,
params: string,
response: string,
): Promise<void> {
if (!this.chatSessionId) {
this.log.warn("toolCall event received but no chat session is active");
return;
}
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onToolCall(callId, name, params, response);
if (this.currentTurnId) {
// Flush current buffer before adding tool block
await this.flushBuffer(this.currentTurnId);
// Add tool block immediately
const turn = await ChatTurn.findById(this.currentTurnId);
if (turn) {
turn.blocks.push({
mode: 'tool',
createdAt: new Date(),
content: {
callId,
name,
parameters: params,
response,
},
});
turn.toolCalls.push({
callId,
name,
parameters: params,
response,
});
turn.stats.toolCallCount = turn.toolCalls.length;
await turn.save();
}
}
} catch (error) {
this.log.error("failed to route toolCall event", { error });
}
}
/**
* Called when the drone completes a work order.
* Flushes any remaining buffered content before finalizing.
*/
async onWorkOrderComplete(
turnId: string,
success: boolean,
message?: string,
): Promise<void> {
if (!this.chatSessionId) {
this.log.warn(
"workOrderComplete event received but no chat session is active",
);
return;
}
try {
// Flush any remaining buffered content
await this.flushBuffer(turnId);
this.streamingBuffers.delete(turnId);
const turn = await ChatTurn.findById(turnId);
if (turn) {
turn.status = success ? ChatTurnStatus.Finished : ChatTurnStatus.Error;
if (!success && message) {
turn.errorMessage = message;
}
await turn.save();
}
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onWorkOrderComplete(turnId, success, message);
this.currentTurnId = undefined;
} catch (error) {
this.log.error("failed to process workOrderComplete event", { error });
}
}
/**
* Sets the active chat session ID for this drone session.
*/
setChatSessionId(chatSessionId: GadgetId): void {
this.chatSessionId = chatSessionId;
// Clear buffer for this turn if exists
this.streamingBuffers.clear();
}
/**
* Gets or creates a streaming buffer for a turn.
*/
private getOrCreateBuffer(turnId: string): IStreamingBuffer {
if (!this.streamingBuffers.has(turnId)) {
this.streamingBuffers.set(turnId, {
currentMode: null,
thinkingContent: '',
respondingContent: '',
});
}
return this.streamingBuffers.get(turnId)!;
}
/**
* Flushes the current buffer to the database.
*/
private async flushBuffer(turnId: string): Promise<void> {
const buffer = this.streamingBuffers.get(turnId);
if (!buffer) return;
const turn = await ChatTurn.findById(turnId);
if (!turn) return;
// Flush thinking content
if (buffer.currentMode === 'thinking' && buffer.thinkingContent) {
turn.blocks.push({
mode: 'thinking',
createdAt: buffer.lastBlockCreatedAt || new Date(),
content: buffer.thinkingContent,
});
buffer.thinkingContent = '';
}
// Flush responding content
if (buffer.currentMode === 'responding' && buffer.respondingContent) {
turn.blocks.push({
mode: 'responding',
createdAt: buffer.lastBlockCreatedAt || new Date(),
content: buffer.respondingContent,
});
buffer.respondingContent = '';
}
if (turn.blocks.length > 0) {
await turn.save();
}
}
/**
* Sets the current turn ID being processed by this drone.
*/
setCurrentTurnId(turnId: GadgetId): void {
this.currentTurnId = turnId;
}
/**
* Called when the drone emits a workspace mode change.
*/
async onWorkspaceModeChanged(mode: WorkspaceMode): Promise<void> {
if (!this.chatSessionId) {
return;
}
this.workspaceMode = mode;
this.log.info("workspace mode changed", { mode });
const codeSession = SocketService.getCodeSessionByChatSessionId(
this.chatSessionId,
);
codeSession.onWorkspaceModeChanged(mode);
}
async onAgentThinking(data: { agentId: string; thinking: string }): Promise<void> {
if (!this.chatSessionId) return;
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(this.chatSessionId);
codeSession.socket.emit("agent:thinking", data);
} catch (error) {
this.log.error("failed to route agent:thinking", { error });
}
}
async onAgentResponse(data: { agentId: string; chunk: string }): Promise<void> {
if (!this.chatSessionId) return;
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(this.chatSessionId);
codeSession.socket.emit("agent:response", data);
} catch (error) {
this.log.error("failed to route agent:response", { error });
}
}
async onAgentToolCall(data: { agentId: string; tool: string; args: unknown }): Promise<void> {
if (!this.chatSessionId) return;
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(this.chatSessionId);
codeSession.socket.emit("agent:tool-call", data);
} catch (error) {
this.log.error("failed to route agent:tool-call", { error });
}
}
async onAgentToolResult(data: { agentId: string; tool: string; result: unknown }): Promise<void> {
if (!this.chatSessionId) return;
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(this.chatSessionId);
codeSession.socket.emit("agent:tool-result", data);
} catch (error) {
this.log.error("failed to route agent:tool-result", { error });
}
}
async onAgentComplete(data: { agentId: string; response?: string; subagent?: Record<string, unknown>; stats?: Record<string, unknown> }): Promise<void> {
if (!this.chatSessionId) return;
try {
const codeSession = SocketService.getCodeSessionByChatSessionId(this.chatSessionId);
codeSession.socket.emit("agent:complete", data);
} catch (error) {
this.log.error("failed to route agent:complete to frontend", { error });
}
// Update the persisted tool call with the final response and subagent data
if (this.currentTurnId && data.agentId) {
try {
const turn = await ChatTurn.findById(this.currentTurnId);
if (turn) {
const toolCall = turn.toolCalls.find((tc: IChatToolCall) => tc.callId === data.agentId);
if (toolCall) {
if (data.response) toolCall.response = data.response;
if (data.subagent) (toolCall as any).subagent = data.subagent;
}
const block = turn.blocks.find((b: IChatTurnBlock) => b.mode === 'tool' && (b.content as IChatToolCall).callId === data.agentId);
if (block && block.mode === 'tool') {
if (data.response) (block.content as IChatToolCall).response = data.response;
if (data.subagent) (block.content as any).subagent = data.subagent;
}
await turn.save();
}
} catch (error) {
this.log.error("failed to update subagent tool call in DB", { error });
}
}
}
/**
* Called when the drone requests crash recovery for an incomplete work order.
*/
async onRequestCrashRecovery(data: {
workspaceId: string;
turnId: string;
chatSessionId: string;
}): Promise<void> {
this.log.info("crash recovery request received", {
workspaceId: data.workspaceId,
turnId: data.turnId,
});
try {
const turn = await ChatTurn.findById(data.turnId);
if (!turn) {
this.log.warn("crash recovery: turn not found", {
turnId: data.turnId,
});
this.socket.emit("crashRecoveryResponse", {
turnId: data.turnId,
action: "discard",
});
return;
}
if (turn.status === ChatTurnStatus.Finished) {
this.log.info("crash recovery: turn already finished", {
turnId: data.turnId,
});
this.socket.emit("crashRecoveryResponse", {
turnId: data.turnId,
action: "discard",
});
return;
}
// Turn is still processing - mark for retry
turn.status = ChatTurnStatus.Error;
turn.errorMessage = "Drone crashed during processing - retrying";
await turn.save();
this.socket.emit("crashRecoveryResponse", {
turnId: data.turnId,
action: "retry",
retryDelay: 5000,
});
this.log.info("crash recovery: scheduled retry", {
turnId: data.turnId,
});
// Schedule retry (will route to same workspaceId)
setTimeout(() => {
this.retryWorkOrder(turn);
}, 5000);
} catch (error) {
const err = error as Error;
this.log.error("crash recovery failed", {
error: err.message,
});
this.socket.emit("crashRecoveryResponse", {
turnId: data.turnId,
action: "discard",
});
}
}
/**
* Retries a work order after crash recovery.
*/
private async retryWorkOrder(turn: any): Promise<void> {
// TODO: Re-emit processWorkOrder to this drone
this.log.info("work order retry not yet implemented", {
turnId: turn._id,
});
}
/**
* Called when the platform requests termination of this drone.
* Forwards the termination request to the drone socket with logging.
* @param cb Callback to invoke with termination result
*/
async onRequestTermination(cb: (success: boolean) => void): Promise<void> {
this.log.info("requestTermination received, forwarding to drone", {
registrationId: this.registration._id,
});
this.socket.emit("requestTermination", (success: boolean) => {
this.log.info("requestTermination forwarded to drone", { success });
cb(success);
});
}
}