rspace-online/lib/mi-actions.ts

137 lines
5.1 KiB
TypeScript

/**
* MI Action Protocol — types and parser for [MI_ACTION:{...}] markers
* embedded in LLM responses.
*
* The LLM outputs prose interleaved with action markers. The parser
* extracts actions and returns clean display text for the user.
*/
export type MiAction =
// Canvas shape actions (existing)
| { type: "create-shape"; tagName: string; props: Record<string, any>; ref?: string }
| { type: "update-shape"; shapeId: string; fields: Record<string, any> }
| { type: "delete-shape"; shapeId: string }
| { type: "connect"; sourceId: string; targetId: string; color?: string }
| { type: "move-shape"; shapeId: string; x: number; y: number }
| { type: "navigate"; path: string }
| { type: "transform"; transform: string; shapeIds: string[] }
// Module content actions
| { type: "create-content"; module: string; contentType: string; body: Record<string, any>; ref?: string }
| { type: "update-content"; module: string; contentType: string; id: string; body: Record<string, any> }
| { type: "delete-content"; module: string; contentType: string; id: string }
// Admin actions
| { type: "enable-module"; moduleId: string }
| { type: "disable-module"; moduleId: string }
| { type: "configure-module"; moduleId: string; settings: Record<string, any> }
// Composite actions
| { type: "scaffold"; name: string; steps: MiAction[] }
| { type: "batch"; actions: MiAction[]; requireConfirm?: boolean };
export interface ParsedMiResponse {
displayText: string;
actions: MiAction[];
}
const ACTION_PATTERN = /\[MI_ACTION:([\s\S]*?)\]/g;
/**
* Parse [MI_ACTION:{...}] markers from streamed text.
* Returns the clean display text (markers stripped) and an array of actions.
*/
export function parseMiActions(text: string): ParsedMiResponse {
const actions: MiAction[] = [];
const displayText = text.replace(ACTION_PATTERN, (_, json) => {
try {
const action = JSON.parse(json.trim()) as MiAction;
if (action && action.type) {
actions.push(action);
}
} catch {
// Malformed action — skip silently
}
return "";
});
return {
displayText: displayText.replace(/\n{3,}/g, "\n\n").trim(),
actions,
};
}
/** Check if an action requires user confirmation before executing. */
export function isDestructiveAction(action: MiAction): boolean {
switch (action.type) {
case "delete-shape":
case "delete-content":
case "disable-module":
return true;
case "batch":
return action.requireConfirm || action.actions.some(isDestructiveAction);
case "scaffold":
return action.steps.some(isDestructiveAction);
default:
return false;
}
}
/** Summarise executed actions for a confirmation chip. */
export function summariseActions(actions: MiAction[]): string {
const counts: Record<string, number> = {};
for (const a of actions) {
counts[a.type] = (counts[a.type] || 0) + 1;
}
const parts: string[] = [];
if (counts["create-shape"]) parts.push(`Created ${counts["create-shape"]} shape(s)`);
if (counts["update-shape"]) parts.push(`Updated ${counts["update-shape"]} shape(s)`);
if (counts["delete-shape"]) parts.push(`Deleted ${counts["delete-shape"]} shape(s)`);
if (counts["connect"]) parts.push(`Connected ${counts["connect"]} pair(s)`);
if (counts["move-shape"]) parts.push(`Moved ${counts["move-shape"]} shape(s)`);
if (counts["transform"]) parts.push(`Applied ${counts["transform"]} transform(s)`);
if (counts["navigate"]) parts.push(`Navigating`);
if (counts["create-content"]) parts.push(`Created ${counts["create-content"]} item(s)`);
if (counts["update-content"]) parts.push(`Updated ${counts["update-content"]} item(s)`);
if (counts["delete-content"]) parts.push(`Deleted ${counts["delete-content"]} item(s)`);
if (counts["enable-module"]) parts.push(`Enabled ${counts["enable-module"]} module(s)`);
if (counts["disable-module"]) parts.push(`Disabled ${counts["disable-module"]} module(s)`);
if (counts["configure-module"]) parts.push(`Configured ${counts["configure-module"]} module(s)`);
if (counts["scaffold"]) parts.push(`Scaffolded ${counts["scaffold"]} setup(s)`);
if (counts["batch"]) parts.push(`Batch: ${counts["batch"]} group(s)`);
return parts.join(", ") || "";
}
/** Produce a detailed list of actions for the expandable summary. */
export function detailedActionSummary(actions: MiAction[]): string[] {
const details: string[] = [];
for (const a of actions) {
switch (a.type) {
case "create-shape":
details.push(`Created ${a.tagName}${a.props?.title ? `: "${a.props.title}"` : ""}`);
break;
case "create-content":
details.push(`Created ${a.contentType} in ${a.module}${a.body?.title ? `: "${a.body.title}"` : ""}`);
break;
case "delete-shape":
details.push(`Deleted shape ${a.shapeId}`);
break;
case "delete-content":
details.push(`Deleted ${a.contentType} ${a.id} from ${a.module}`);
break;
case "connect":
details.push(`Connected ${a.sourceId}${a.targetId}`);
break;
case "scaffold":
details.push(`Scaffold "${a.name}": ${a.steps.length} steps`);
break;
case "batch":
details.push(`Batch: ${a.actions.length} actions`);
break;
case "navigate":
details.push(`Navigate to ${a.path}`);
break;
default:
details.push(`${a.type}`);
}
}
return details;
}