@robota-sdk/agent-transport 3.0.0-beta.64

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (183) hide show
  1. package/LICENSE +21 -0
  2. package/dist/node/headless/index.cjs +1 -0
  3. package/dist/node/headless/index.d.ts +2 -0
  4. package/dist/node/headless/index.js +1 -0
  5. package/dist/node/headless-CWEpJXFK.js +7 -0
  6. package/dist/node/headless-CWEpJXFK.js.map +1 -0
  7. package/dist/node/headless-CsZFelG9.cjs +6 -0
  8. package/dist/node/http/index.cjs +1 -0
  9. package/dist/node/http/index.d.ts +2 -0
  10. package/dist/node/http/index.js +1 -0
  11. package/dist/node/http-CM3TJhrF.cjs +1 -0
  12. package/dist/node/http-DwO1AHG-.js +2 -0
  13. package/dist/node/http-DwO1AHG-.js.map +1 -0
  14. package/dist/node/index--Ti9NzQX.d.ts +64 -0
  15. package/dist/node/index--Ti9NzQX.d.ts.map +1 -0
  16. package/dist/node/index-B_rcr14p.d.ts +47 -0
  17. package/dist/node/index-B_rcr14p.d.ts.map +1 -0
  18. package/dist/node/index-C9LWCL4l.d.ts +34 -0
  19. package/dist/node/index-C9LWCL4l.d.ts.map +1 -0
  20. package/dist/node/index-CAr3ioVh.d.ts +64 -0
  21. package/dist/node/index-CAr3ioVh.d.ts.map +1 -0
  22. package/dist/node/index-CEs25wVk.d.ts +213 -0
  23. package/dist/node/index-CEs25wVk.d.ts.map +1 -0
  24. package/dist/node/index-CvXLpjJO.d.ts +213 -0
  25. package/dist/node/index-CvXLpjJO.d.ts.map +1 -0
  26. package/dist/node/index-D34WUfFH.d.ts +26 -0
  27. package/dist/node/index-D34WUfFH.d.ts.map +1 -0
  28. package/dist/node/index-Y0zHb1Bz.d.ts +47 -0
  29. package/dist/node/index-Y0zHb1Bz.d.ts.map +1 -0
  30. package/dist/node/index-k3TUjA-T.d.ts +26 -0
  31. package/dist/node/index-k3TUjA-T.d.ts.map +1 -0
  32. package/dist/node/index-nBlMTFkZ.d.ts +34 -0
  33. package/dist/node/index-nBlMTFkZ.d.ts.map +1 -0
  34. package/dist/node/index.cjs +1 -0
  35. package/dist/node/index.d.ts +6 -0
  36. package/dist/node/index.js +1 -0
  37. package/dist/node/mcp/index.cjs +1 -0
  38. package/dist/node/mcp/index.d.ts +2 -0
  39. package/dist/node/mcp/index.js +1 -0
  40. package/dist/node/mcp-BXBwF6Wu.js +2 -0
  41. package/dist/node/mcp-BXBwF6Wu.js.map +1 -0
  42. package/dist/node/mcp-DcHuGokt.cjs +1 -0
  43. package/dist/node/tui/index.cjs +1 -0
  44. package/dist/node/tui/index.d.ts +2 -0
  45. package/dist/node/tui/index.js +1 -0
  46. package/dist/node/tui-CeD_6rSo.cjs +24 -0
  47. package/dist/node/tui-zmDTPk4b.js +25 -0
  48. package/dist/node/tui-zmDTPk4b.js.map +1 -0
  49. package/dist/node/ws/index.cjs +1 -0
  50. package/dist/node/ws/index.d.ts +2 -0
  51. package/dist/node/ws/index.js +1 -0
  52. package/dist/node/ws-B-oRccFl.js +2 -0
  53. package/dist/node/ws-B-oRccFl.js.map +1 -0
  54. package/dist/node/ws-COnIgnmn.cjs +1 -0
  55. package/package.json +141 -0
  56. package/src/headless/__tests__/headless-runner-initialization.test.ts +45 -0
  57. package/src/headless/__tests__/headless-runner.test.ts +484 -0
  58. package/src/headless/__tests__/headless-skill-activation.integration.test.ts +430 -0
  59. package/src/headless/__tests__/headless-transport.test.ts +268 -0
  60. package/src/headless/headless-runner.ts +141 -0
  61. package/src/headless/headless-stream-json.ts +142 -0
  62. package/src/headless/headless-transport.ts +43 -0
  63. package/src/headless/index.ts +4 -0
  64. package/src/http/__tests__/http-transport.test.ts +55 -0
  65. package/src/http/__tests__/routes.test.ts +168 -0
  66. package/src/http/http-transport.ts +42 -0
  67. package/src/http/index.ts +4 -0
  68. package/src/http/routes.ts +151 -0
  69. package/src/index.ts +5 -0
  70. package/src/mcp/__tests__/mcp-server.test.ts +66 -0
  71. package/src/mcp/__tests__/mcp-transport.test.ts +46 -0
  72. package/src/mcp/index.ts +4 -0
  73. package/src/mcp/mcp-server.ts +162 -0
  74. package/src/mcp/mcp-transport.ts +48 -0
  75. package/src/tui/App.tsx +478 -0
  76. package/src/tui/BackgroundTaskPanel.tsx +34 -0
  77. package/src/tui/CjkTextInput.tsx +204 -0
  78. package/src/tui/ConfirmPrompt.tsx +69 -0
  79. package/src/tui/ExecutionWorkspaceDetailPane.tsx +62 -0
  80. package/src/tui/ExecutionWorkspaceSwitcher.tsx +185 -0
  81. package/src/tui/InkTerminal.ts +42 -0
  82. package/src/tui/InputArea.tsx +298 -0
  83. package/src/tui/InteractivePrompt.tsx +57 -0
  84. package/src/tui/ListPicker.tsx +94 -0
  85. package/src/tui/MenuSelect.tsx +103 -0
  86. package/src/tui/MessageList.tsx +282 -0
  87. package/src/tui/PermissionPrompt.tsx +84 -0
  88. package/src/tui/PluginTUI.tsx +256 -0
  89. package/src/tui/SessionPicker.tsx +66 -0
  90. package/src/tui/SessionStatusBar.tsx +66 -0
  91. package/src/tui/SlashAutocomplete.tsx +110 -0
  92. package/src/tui/StatusBar.tsx +213 -0
  93. package/src/tui/StreamingIndicator.tsx +91 -0
  94. package/src/tui/TextPrompt.tsx +80 -0
  95. package/src/tui/ToolCommandOutput.tsx +37 -0
  96. package/src/tui/ToolDiffBlock.tsx +30 -0
  97. package/src/tui/TransportTUI.tsx +116 -0
  98. package/src/tui/UpdateNotice.tsx +14 -0
  99. package/src/tui/UsageSummaryEntry.tsx +38 -0
  100. package/src/tui/WaveText.tsx +44 -0
  101. package/src/tui/__tests__/InteractivePrompt.test.tsx +82 -0
  102. package/src/tui/__tests__/ListPicker.test.tsx +159 -0
  103. package/src/tui/__tests__/MenuSelect.test.tsx +103 -0
  104. package/src/tui/__tests__/PluginTUI.test.tsx +167 -0
  105. package/src/tui/__tests__/SlashAutocomplete.test.tsx +140 -0
  106. package/src/tui/__tests__/TextPrompt.test.tsx +98 -0
  107. package/src/tui/__tests__/UpdateNotice.test.tsx +15 -0
  108. package/src/tui/__tests__/abort-after-permission.test.tsx +169 -0
  109. package/src/tui/__tests__/abort-streaming-e2e.test.tsx +183 -0
  110. package/src/tui/__tests__/background-task-panel.test.tsx +53 -0
  111. package/src/tui/__tests__/background-task-row-format.test.ts +59 -0
  112. package/src/tui/__tests__/cjk-text-input-flow.test.ts +109 -0
  113. package/src/tui/__tests__/cjk-text-input.test.ts +191 -0
  114. package/src/tui/__tests__/command-effect-handler.test.ts +128 -0
  115. package/src/tui/__tests__/command-output-summary.test.ts +95 -0
  116. package/src/tui/__tests__/compact-event-bridge.test.ts +20 -0
  117. package/src/tui/__tests__/confirm-permission-flow.test.ts +91 -0
  118. package/src/tui/__tests__/confirm-prompt.test.tsx +87 -0
  119. package/src/tui/__tests__/execution-workspace-switcher.test.tsx +110 -0
  120. package/src/tui/__tests__/execution-workspace-view-model.test.ts +93 -0
  121. package/src/tui/__tests__/fixtures/provider-setup-prompt-driver.tsx +122 -0
  122. package/src/tui/__tests__/input-area-flow.test.ts +152 -0
  123. package/src/tui/__tests__/message-list-rendering.test.tsx +353 -0
  124. package/src/tui/__tests__/model-change-side-effect.test.ts +91 -0
  125. package/src/tui/__tests__/prompt-queue.test.tsx +255 -0
  126. package/src/tui/__tests__/provider-setup-pty-e2e.test.ts +233 -0
  127. package/src/tui/__tests__/render-markdown.test.ts +72 -0
  128. package/src/tui/__tests__/selection-flow.test.ts +61 -0
  129. package/src/tui/__tests__/slash-routing-effects.test.ts +225 -0
  130. package/src/tui/__tests__/status-activity.test.ts +71 -0
  131. package/src/tui/__tests__/status-bar.test.tsx +157 -0
  132. package/src/tui/__tests__/streaming-indicator.test.tsx +137 -0
  133. package/src/tui/__tests__/text-prompt-flow.test.ts +77 -0
  134. package/src/tui/__tests__/tui-state-manager.test.ts +401 -0
  135. package/src/tui/background-task-row-format.ts +52 -0
  136. package/src/tui/command-output-summary.ts +122 -0
  137. package/src/tui/execution-workspace-view-model.ts +123 -0
  138. package/src/tui/flows/cjk-text-input-flow.ts +285 -0
  139. package/src/tui/flows/confirm-prompt-flow.ts +45 -0
  140. package/src/tui/flows/input-area-flow.ts +186 -0
  141. package/src/tui/flows/permission-prompt-flow.ts +76 -0
  142. package/src/tui/flows/selection-flow.ts +126 -0
  143. package/src/tui/flows/text-prompt-flow.ts +98 -0
  144. package/src/tui/hooks/command-effect-handler.ts +98 -0
  145. package/src/tui/hooks/command-effect-queue.ts +39 -0
  146. package/src/tui/hooks/model-change-side-effect.ts +63 -0
  147. package/src/tui/hooks/side-effects-types.ts +38 -0
  148. package/src/tui/hooks/use-interactive-session-init.ts +50 -0
  149. package/src/tui/hooks/useAutocomplete.ts +85 -0
  150. package/src/tui/hooks/useInteractiveSession.ts +273 -0
  151. package/src/tui/hooks/usePermissionQueue.ts +51 -0
  152. package/src/tui/hooks/usePluginCallbacks.ts +30 -0
  153. package/src/tui/hooks/usePluginScreenData.ts +84 -0
  154. package/src/tui/hooks/useSideEffects.ts +210 -0
  155. package/src/tui/hooks/useSlashRouting.ts +117 -0
  156. package/src/tui/hooks/useStatusLineSettings.ts +35 -0
  157. package/src/tui/index.ts +3 -0
  158. package/src/tui/plugin-tui-handlers.ts +163 -0
  159. package/src/tui/render-markdown.ts +129 -0
  160. package/src/tui/render.tsx +60 -0
  161. package/src/tui/status-activity.ts +63 -0
  162. package/src/tui/tui-cli-adapter-context.tsx +12 -0
  163. package/src/tui/tui-cli-adapter.ts +25 -0
  164. package/src/tui/tui-state-manager.ts +225 -0
  165. package/src/tui/tui-transport.ts +32 -0
  166. package/src/tui/types.ts +14 -0
  167. package/src/tui/utils/__tests__/edit-diff.test.ts +426 -0
  168. package/src/tui/utils/__tests__/paste-detection.test.ts +116 -0
  169. package/src/tui/utils/__tests__/paste-labels.test.ts +46 -0
  170. package/src/tui/utils/__tests__/tool-call-extractor.test.ts +227 -0
  171. package/src/tui/utils/__tests__/tool-diff-summary.test.ts +104 -0
  172. package/src/tui/utils/edit-diff.ts +152 -0
  173. package/src/tui/utils/paste-labels.ts +9 -0
  174. package/src/tui/utils/tool-call-extractor.ts +91 -0
  175. package/src/tui/utils/tool-diff-summary.ts +75 -0
  176. package/src/ws/__tests__/ws-handler.test.ts +407 -0
  177. package/src/ws/__tests__/ws-transport.test.ts +53 -0
  178. package/src/ws/index.ts +13 -0
  179. package/src/ws/ws-background-messages.ts +170 -0
  180. package/src/ws/ws-handler.ts +279 -0
  181. package/src/ws/ws-protocol.ts +76 -0
  182. package/src/ws/ws-transport-configurable.ts +123 -0
  183. package/src/ws/ws-transport.ts +42 -0
@@ -0,0 +1,91 @@
1
+ /**
2
+ * Extracts tool call summaries from session history messages.
3
+ * Pure function — no side effects, no framework dependencies.
4
+ */
5
+
6
+ import { extractEditDiff } from './edit-diff.js';
7
+ import type { IDiffLine } from './edit-diff.js';
8
+ import type { TUniversalValue } from '@robota-sdk/agent-core';
9
+
10
+ const TOOL_ARG_MAX_LENGTH = 80;
11
+ const TAIL_KEEP = 30;
12
+ const ELLIPSIS_LEN = 3;
13
+
14
+ interface IHistoryMessage {
15
+ role: string;
16
+ toolCalls?: Array<{
17
+ function: { name: string; arguments: string };
18
+ }>;
19
+ }
20
+
21
+ /** A tool call summary with optional diff for Edit tools */
22
+ export interface IToolCallSummary {
23
+ line: string;
24
+ diffLines?: IDiffLine[];
25
+ diffFile?: string;
26
+ }
27
+
28
+ /**
29
+ * Extract tool call display lines from history messages.
30
+ * Format: `ToolName(firstArgValue)` — first argument value truncated to 80 chars.
31
+ * Edit tools include diff information.
32
+ */
33
+ export function extractToolCalls(history: IHistoryMessage[], startIndex: number): string[] {
34
+ return extractToolCallsWithDiff(history, startIndex).map((s) => s.line);
35
+ }
36
+
37
+ /**
38
+ * Extract tool call summaries with diff info from history messages.
39
+ */
40
+ export function extractToolCallsWithDiff(
41
+ history: IHistoryMessage[],
42
+ startIndex: number,
43
+ ): IToolCallSummary[] {
44
+ const summaries: IToolCallSummary[] = [];
45
+ for (let i = startIndex; i < history.length; i++) {
46
+ const msg = history[i];
47
+ if (msg.role === 'assistant' && msg.toolCalls) {
48
+ for (const tc of msg.toolCalls) {
49
+ const value = parseFirstArgValue(tc.function.arguments);
50
+ const truncated =
51
+ value.length > TOOL_ARG_MAX_LENGTH
52
+ ? value.slice(0, TOOL_ARG_MAX_LENGTH - TAIL_KEEP - ELLIPSIS_LEN) +
53
+ '...' +
54
+ value.slice(-TAIL_KEEP)
55
+ : value;
56
+
57
+ const summary: IToolCallSummary = {
58
+ line: `${tc.function.name}(${truncated})`,
59
+ };
60
+
61
+ // Extract diff for Edit tool
62
+ if (tc.function.name === 'Edit') {
63
+ try {
64
+ const args = JSON.parse(tc.function.arguments) as Record<string, TUniversalValue>;
65
+ const diff = extractEditDiff('Edit', args);
66
+ if (diff) {
67
+ summary.diffLines = diff.lines;
68
+ summary.diffFile = diff.file;
69
+ }
70
+ } catch {
71
+ // ignore parse errors
72
+ }
73
+ }
74
+
75
+ summaries.push(summary);
76
+ }
77
+ }
78
+ }
79
+ return summaries;
80
+ }
81
+
82
+ /** Parse the first argument value from a JSON arguments string. */
83
+ function parseFirstArgValue(argsJson: string): string {
84
+ try {
85
+ const parsed = JSON.parse(argsJson) as Record<string, TUniversalValue>;
86
+ const firstVal = Object.values(parsed)[0];
87
+ return typeof firstVal === 'string' ? firstVal : JSON.stringify(firstVal);
88
+ } catch {
89
+ return argsJson;
90
+ }
91
+ }
@@ -0,0 +1,75 @@
1
+ import type { IDiffLine } from './edit-diff.js';
2
+
3
+ const MAX_DIFF_LINES = 12;
4
+ const TRUNCATED_SHOW = 10;
5
+
6
+ export interface IToolDiffSummaryInput {
7
+ file?: string;
8
+ lines: readonly IDiffLine[];
9
+ }
10
+
11
+ export interface IToolDiffSummary {
12
+ file?: string;
13
+ markdown: string;
14
+ truncated: boolean;
15
+ remainingLineCount: number;
16
+ }
17
+
18
+ export function buildToolDiffSummary(input: IToolDiffSummaryInput): IToolDiffSummary {
19
+ const visibleLines =
20
+ input.lines.length > MAX_DIFF_LINES ? selectVisibleDiffLines(input.lines) : input.lines;
21
+ const lineNumberWidth = Math.max(...visibleLines.map((line) => line.lineNumber), 0).toString()
22
+ .length;
23
+ const body = visibleLines.map((line) => formatDiffLine(line, lineNumberWidth));
24
+ const truncated = input.lines.length > MAX_DIFF_LINES;
25
+
26
+ return {
27
+ file: input.file,
28
+ markdown: ['```diff', ...body, '```'].join('\n'),
29
+ truncated,
30
+ remainingLineCount: truncated ? input.lines.length - visibleLines.length : 0,
31
+ };
32
+ }
33
+
34
+ function formatDiffLine(line: IDiffLine, lineNumberWidth: number): string {
35
+ if (line.type === 'hunk') return line.text;
36
+ const lineNumber = line.lineNumber.toString().padStart(lineNumberWidth, ' ');
37
+ if (line.type === 'remove') return `- ${lineNumber} | ${line.text}`;
38
+ if (line.type === 'add') return `+ ${lineNumber} | ${line.text}`;
39
+ return ` ${lineNumber} | ${line.text}`;
40
+ }
41
+
42
+ function selectVisibleDiffLines(lines: readonly IDiffLine[]): readonly IDiffLine[] {
43
+ const groups = groupByHunk(lines);
44
+ const visible: IDiffLine[] = [];
45
+
46
+ for (const group of groups) {
47
+ if (visible.length === 0 && group.length > TRUNCATED_SHOW) {
48
+ return group.slice(0, TRUNCATED_SHOW);
49
+ }
50
+ if (visible.length + group.length > TRUNCATED_SHOW) break;
51
+ visible.push(...group);
52
+ }
53
+
54
+ return visible.length > 0 ? visible : lines.slice(0, TRUNCATED_SHOW);
55
+ }
56
+
57
+ function groupByHunk(lines: readonly IDiffLine[]): IDiffLine[][] {
58
+ const groups: IDiffLine[][] = [];
59
+ let current: IDiffLine[] = [];
60
+
61
+ for (const line of lines) {
62
+ if (line.type === 'hunk' && current.length > 0) {
63
+ groups.push(current);
64
+ current = [line];
65
+ continue;
66
+ }
67
+ current.push(line);
68
+ }
69
+
70
+ if (current.length > 0) {
71
+ groups.push(current);
72
+ }
73
+
74
+ return groups;
75
+ }
@@ -0,0 +1,407 @@
1
+ /**
2
+ * Tests for WebSocket transport handler.
3
+ */
4
+
5
+ import { describe, it, expect, vi } from 'vitest';
6
+ import { createWsHandler } from '../ws-handler.js';
7
+ import type { TServerMessage } from '../ws-protocol.js';
8
+ import type {
9
+ IBackgroundTaskLogPage,
10
+ IBackgroundTaskState,
11
+ IExecutionWorkspaceEvent,
12
+ IExecutionWorkspaceSnapshot,
13
+ IInteractiveSession,
14
+ IBackgroundJobGroupState,
15
+ TBackgroundJobGroupEvent,
16
+ TBackgroundTaskEvent,
17
+ TExecutionWorkspaceUpdateCause,
18
+ } from '@robota-sdk/agent-framework';
19
+
20
+ const backgroundTask: IBackgroundTaskState = {
21
+ id: 'task_1',
22
+ kind: 'agent',
23
+ label: 'Explore',
24
+ agentType: 'Explore',
25
+ status: 'running',
26
+ mode: 'background',
27
+ parentSessionId: 'session_1',
28
+ depth: 0,
29
+ cwd: '/repo',
30
+ updatedAt: '2026-05-01T00:00:00.000Z',
31
+ unread: false,
32
+ promptPreview: 'inspect code',
33
+ };
34
+
35
+ const backgroundTaskLogPage: IBackgroundTaskLogPage = {
36
+ taskId: 'task_1',
37
+ cursor: { offset: 0 },
38
+ lines: ['line one'],
39
+ };
40
+
41
+ const executionWorkspaceSnapshot: IExecutionWorkspaceSnapshot = {
42
+ sessionId: 'session_1',
43
+ entries: [],
44
+ updatedAt: '2026-05-01T00:00:00.000Z',
45
+ };
46
+
47
+ const backgroundJobGroup: IBackgroundJobGroupState = {
48
+ id: 'group_1',
49
+ parentSessionId: 'session_1',
50
+ waitPolicy: 'wait_all',
51
+ taskIds: ['task_1'],
52
+ status: 'running',
53
+ createdAt: '2026-05-01T00:00:00.000Z',
54
+ updatedAt: '2026-05-01T00:00:00.000Z',
55
+ results: [],
56
+ };
57
+
58
+ function createMockSession() {
59
+ const listeners = new Map<string, Set<(...args: unknown[]) => void>>();
60
+ return {
61
+ submit: vi.fn(),
62
+ abort: vi.fn(),
63
+ cancelQueue: vi.fn(),
64
+ getMessages: vi.fn().mockReturnValue([{ role: 'user', content: 'hi' }]),
65
+ getContextState: vi
66
+ .fn()
67
+ .mockReturnValue({ usedTokens: 500, maxTokens: 100000, usedPercentage: 0.5 }),
68
+ isExecuting: vi.fn().mockReturnValue(false),
69
+ getPendingPrompt: vi.fn().mockReturnValue(null),
70
+ listBackgroundTasks: vi.fn().mockReturnValue([backgroundTask]),
71
+ getBackgroundTask: vi.fn().mockReturnValue(backgroundTask),
72
+ listBackgroundJobGroups: vi.fn().mockReturnValue([backgroundJobGroup]),
73
+ getBackgroundJobGroup: vi.fn().mockReturnValue(backgroundJobGroup),
74
+ waitBackgroundJobGroup: vi.fn().mockResolvedValue({
75
+ ...backgroundJobGroup,
76
+ status: 'completed',
77
+ completedAt: '2026-05-01T00:00:01.000Z',
78
+ results: [{ taskId: 'task_1', label: 'Explore', status: 'completed', summary: 'done' }],
79
+ }),
80
+ cancelBackgroundTask: vi.fn().mockResolvedValue(undefined),
81
+ closeBackgroundTask: vi.fn().mockResolvedValue(undefined),
82
+ sendBackgroundTask: vi.fn().mockResolvedValue(undefined),
83
+ readBackgroundTaskLog: vi.fn().mockResolvedValue(backgroundTaskLogPage),
84
+ getExecutionWorkspaceSnapshot: vi.fn().mockReturnValue(executionWorkspaceSnapshot),
85
+ executeCommand: vi.fn().mockResolvedValue({ message: 'done', success: true, data: {} }),
86
+ listCommands: vi.fn().mockReturnValue([]),
87
+ on: vi.fn((event: string, handler: (...args: unknown[]) => void) => {
88
+ if (!listeners.has(event)) listeners.set(event, new Set());
89
+ listeners.get(event)!.add(handler);
90
+ }),
91
+ off: vi.fn((event: string, handler: (...args: unknown[]) => void) => {
92
+ listeners.get(event)?.delete(handler);
93
+ }),
94
+ _emit: (event: string, ...args: unknown[]) => {
95
+ listeners.get(event)?.forEach((h) => h(...args));
96
+ },
97
+ } as unknown as IInteractiveSession & { _emit: (event: string, ...args: unknown[]) => void };
98
+ }
99
+
100
+ describe('WebSocket Transport Handler', () => {
101
+ function setup() {
102
+ const session = createMockSession();
103
+ const sent: TServerMessage[] = [];
104
+ const { onMessage, cleanup } = createWsHandler({
105
+ session: session as unknown as IInteractiveSession,
106
+ send: (msg) => sent.push(msg),
107
+ });
108
+ return { session, sent, onMessage, cleanup };
109
+ }
110
+
111
+ it('submit calls session.submit()', () => {
112
+ const { onMessage, session } = setup();
113
+ onMessage(JSON.stringify({ type: 'submit', prompt: 'hello' }));
114
+ expect(
115
+ (session as unknown as { submit: ReturnType<typeof vi.fn> }).submit,
116
+ ).toHaveBeenCalledWith('hello');
117
+ });
118
+
119
+ it('abort calls session.abort()', () => {
120
+ const { onMessage, session } = setup();
121
+ onMessage(JSON.stringify({ type: 'abort' }));
122
+ expect((session as unknown as { abort: ReturnType<typeof vi.fn> }).abort).toHaveBeenCalled();
123
+ });
124
+
125
+ it('cancel-queue calls session.cancelQueue()', () => {
126
+ const { onMessage, session } = setup();
127
+ onMessage(JSON.stringify({ type: 'cancel-queue' }));
128
+ expect(
129
+ (session as unknown as { cancelQueue: ReturnType<typeof vi.fn> }).cancelQueue,
130
+ ).toHaveBeenCalled();
131
+ });
132
+
133
+ it('get-messages sends messages back', () => {
134
+ const { onMessage, sent } = setup();
135
+ onMessage(JSON.stringify({ type: 'get-messages' }));
136
+ expect(sent).toHaveLength(1);
137
+ expect(sent[0]!.type).toBe('messages');
138
+ });
139
+
140
+ it('get-context sends context state back', () => {
141
+ const { onMessage, sent } = setup();
142
+ onMessage(JSON.stringify({ type: 'get-context' }));
143
+ expect(sent).toHaveLength(1);
144
+ expect(sent[0]!.type).toBe('context');
145
+ });
146
+
147
+ it('get-executing sends executing status', () => {
148
+ const { onMessage, sent } = setup();
149
+ onMessage(JSON.stringify({ type: 'get-executing' }));
150
+ expect(sent[0]).toEqual({ type: 'executing', executing: false });
151
+ });
152
+
153
+ it('get-pending sends pending prompt', () => {
154
+ const { onMessage, sent } = setup();
155
+ onMessage(JSON.stringify({ type: 'get-pending' }));
156
+ expect(sent[0]).toEqual({ type: 'pending', pending: null });
157
+ });
158
+
159
+ it('get-background-tasks sends current background task list', () => {
160
+ const { onMessage, sent, session } = setup();
161
+ onMessage(JSON.stringify({ type: 'get-background-tasks', filter: { kind: 'agent' } }));
162
+
163
+ expect(sent[0]).toEqual({ type: 'background_tasks', tasks: [backgroundTask] });
164
+ expect(
165
+ (session as unknown as { listBackgroundTasks: ReturnType<typeof vi.fn> }).listBackgroundTasks,
166
+ ).toHaveBeenCalledWith({ kind: 'agent' });
167
+ });
168
+
169
+ it('get-background-task sends one background task snapshot', () => {
170
+ const { onMessage, sent, session } = setup();
171
+ onMessage(JSON.stringify({ type: 'get-background-task', taskId: 'task_1' }));
172
+
173
+ expect(sent[0]).toEqual({
174
+ type: 'background_task',
175
+ taskId: 'task_1',
176
+ task: backgroundTask,
177
+ });
178
+ expect(
179
+ (session as unknown as { getBackgroundTask: ReturnType<typeof vi.fn> }).getBackgroundTask,
180
+ ).toHaveBeenCalledWith('task_1');
181
+ });
182
+
183
+ it('get-background-job-groups sends group snapshots', () => {
184
+ const { onMessage, sent, session } = setup();
185
+ onMessage(JSON.stringify({ type: 'get-background-job-groups' }));
186
+
187
+ expect(sent[0]).toEqual({ type: 'background_job_groups', groups: [backgroundJobGroup] });
188
+ expect(
189
+ (session as unknown as { listBackgroundJobGroups: ReturnType<typeof vi.fn> })
190
+ .listBackgroundJobGroups,
191
+ ).toHaveBeenCalled();
192
+ });
193
+
194
+ it('get-background-job-group sends one group snapshot', () => {
195
+ const { onMessage, sent, session } = setup();
196
+ onMessage(JSON.stringify({ type: 'get-background-job-group', groupId: 'group_1' }));
197
+
198
+ expect(sent[0]).toEqual({
199
+ type: 'background_job_group',
200
+ groupId: 'group_1',
201
+ group: backgroundJobGroup,
202
+ });
203
+ expect(
204
+ (session as unknown as { getBackgroundJobGroup: ReturnType<typeof vi.fn> })
205
+ .getBackgroundJobGroup,
206
+ ).toHaveBeenCalledWith('group_1');
207
+ });
208
+
209
+ it('wait-background-job-group waits and sends the completed group', async () => {
210
+ const { onMessage, sent, session } = setup();
211
+ onMessage(JSON.stringify({ type: 'wait-background-job-group', groupId: 'group_1' }));
212
+
213
+ await new Promise((r) => setTimeout(r, 10));
214
+
215
+ expect(
216
+ (session as unknown as { waitBackgroundJobGroup: ReturnType<typeof vi.fn> })
217
+ .waitBackgroundJobGroup,
218
+ ).toHaveBeenCalledWith('group_1');
219
+ expect(sent[0]).toMatchObject({
220
+ type: 'background_job_group',
221
+ groupId: 'group_1',
222
+ group: { id: 'group_1', status: 'completed' },
223
+ });
224
+ });
225
+
226
+ it('cancel-background-task maps to session control and emits control result', async () => {
227
+ const { onMessage, sent, session } = setup();
228
+ onMessage(JSON.stringify({ type: 'cancel-background-task', taskId: 'task_1', reason: 'stop' }));
229
+
230
+ await new Promise((r) => setTimeout(r, 10));
231
+
232
+ expect(
233
+ (session as unknown as { cancelBackgroundTask: ReturnType<typeof vi.fn> })
234
+ .cancelBackgroundTask,
235
+ ).toHaveBeenCalledWith('task_1', 'stop');
236
+ expect(sent[0]).toEqual({
237
+ type: 'background_task_control_result',
238
+ action: 'cancel',
239
+ taskId: 'task_1',
240
+ success: true,
241
+ });
242
+ });
243
+
244
+ it('send-background-task maps prompt input to session control', async () => {
245
+ const { onMessage, sent, session } = setup();
246
+ onMessage(
247
+ JSON.stringify({
248
+ type: 'send-background-task',
249
+ taskId: 'task_1',
250
+ input: { prompt: 'continue' },
251
+ }),
252
+ );
253
+
254
+ await new Promise((r) => setTimeout(r, 10));
255
+
256
+ expect(
257
+ (session as unknown as { sendBackgroundTask: ReturnType<typeof vi.fn> }).sendBackgroundTask,
258
+ ).toHaveBeenCalledWith('task_1', { prompt: 'continue' });
259
+ expect(sent[0]).toEqual({
260
+ type: 'background_task_control_result',
261
+ action: 'send',
262
+ taskId: 'task_1',
263
+ success: true,
264
+ });
265
+ });
266
+
267
+ it('read-background-task-log sends log page', async () => {
268
+ const { onMessage, sent, session } = setup();
269
+ onMessage(
270
+ JSON.stringify({
271
+ type: 'read-background-task-log',
272
+ taskId: 'task_1',
273
+ cursor: { offset: 0 },
274
+ }),
275
+ );
276
+
277
+ await new Promise((r) => setTimeout(r, 10));
278
+
279
+ expect(
280
+ (session as unknown as { readBackgroundTaskLog: ReturnType<typeof vi.fn> })
281
+ .readBackgroundTaskLog,
282
+ ).toHaveBeenCalledWith('task_1', { offset: 0 });
283
+ expect(sent[0]).toEqual({
284
+ type: 'background_task_log',
285
+ taskId: 'task_1',
286
+ page: backgroundTaskLogPage,
287
+ });
288
+ });
289
+
290
+ it('command executes via session.executeCommand()', async () => {
291
+ const { onMessage, sent, session } = setup();
292
+ onMessage(JSON.stringify({ type: 'command', name: 'clear' }));
293
+ await new Promise((r) => setTimeout(r, 10));
294
+ expect(sent).toHaveLength(1);
295
+ expect(sent[0]!.type).toBe('command_result');
296
+ expect(
297
+ (session as unknown as { executeCommand: ReturnType<typeof vi.fn> }).executeCommand,
298
+ ).toHaveBeenCalledWith('clear', '');
299
+ });
300
+
301
+ it('invalid JSON sends protocol_error', () => {
302
+ const { onMessage, sent } = setup();
303
+ onMessage('not json');
304
+ expect(sent[0]).toEqual({ type: 'protocol_error', message: 'Invalid JSON' });
305
+ });
306
+
307
+ it('unknown type sends protocol_error', () => {
308
+ const { onMessage, sent } = setup();
309
+ onMessage(JSON.stringify({ type: 'unknown_type' }));
310
+ expect(sent[0]!.type).toBe('protocol_error');
311
+ });
312
+
313
+ it('submit without prompt sends protocol_error', () => {
314
+ const { onMessage, sent } = setup();
315
+ onMessage(JSON.stringify({ type: 'submit' }));
316
+ expect(sent[0]).toEqual({ type: 'protocol_error', message: 'prompt is required' });
317
+ });
318
+
319
+ it('forwards InteractiveSession events to client', () => {
320
+ const { session, sent } = setup();
321
+ (session as unknown as { _emit: (e: string, ...args: unknown[]) => void })._emit(
322
+ 'text_delta',
323
+ 'hello',
324
+ );
325
+ expect(sent).toHaveLength(1);
326
+ expect(sent[0]).toEqual({ type: 'text_delta', delta: 'hello' });
327
+ });
328
+
329
+ it('forwards thinking event', () => {
330
+ const { session, sent } = setup();
331
+ (session as unknown as { _emit: (e: string, ...args: unknown[]) => void })._emit(
332
+ 'thinking',
333
+ true,
334
+ );
335
+ expect(sent[0]).toEqual({ type: 'thinking', isThinking: true });
336
+ });
337
+
338
+ it('forwards background task events to the client', () => {
339
+ const { session, sent } = setup();
340
+ const event: TBackgroundTaskEvent = {
341
+ type: 'background_task_text_delta',
342
+ taskId: 'task_1',
343
+ delta: 'partial',
344
+ };
345
+ (session as unknown as { _emit: (e: string, ...args: unknown[]) => void })._emit(
346
+ 'background_task_event',
347
+ event,
348
+ );
349
+
350
+ expect(sent[0]).toEqual({ type: 'background_task_event', event });
351
+ });
352
+
353
+ it('forwards background job group events to the client', () => {
354
+ const { session, sent } = setup();
355
+ const event: TBackgroundJobGroupEvent = {
356
+ type: 'background_job_group_completed',
357
+ group: { ...backgroundJobGroup, status: 'completed' },
358
+ };
359
+ (session as unknown as { _emit: (e: string, ...args: unknown[]) => void })._emit(
360
+ 'background_job_group_event',
361
+ event,
362
+ );
363
+
364
+ expect(sent[0]).toEqual({ type: 'background_job_group_event', event });
365
+ });
366
+
367
+ it('forwards execution_workspace_event to the client', () => {
368
+ const { session, sent } = setup();
369
+ const event: IExecutionWorkspaceEvent = {
370
+ type: 'execution_workspace_updated',
371
+ cause: 'main_thread',
372
+ snapshot: executionWorkspaceSnapshot,
373
+ };
374
+ (session as unknown as { _emit: (e: string, ...args: unknown[]) => void })._emit(
375
+ 'execution_workspace_event',
376
+ event,
377
+ );
378
+
379
+ expect(sent[0]).toEqual({
380
+ type: 'execution_workspace_event',
381
+ snapshot: executionWorkspaceSnapshot,
382
+ });
383
+ });
384
+
385
+ it('get-execution-workspace sends current snapshot', () => {
386
+ const { onMessage, sent, session } = setup();
387
+ onMessage(JSON.stringify({ type: 'get-execution-workspace' }));
388
+
389
+ expect(sent[0]).toEqual({
390
+ type: 'execution_workspace_event',
391
+ snapshot: executionWorkspaceSnapshot,
392
+ });
393
+ expect(
394
+ (
395
+ session as unknown as {
396
+ getExecutionWorkspaceSnapshot: ReturnType<typeof vi.fn>;
397
+ }
398
+ ).getExecutionWorkspaceSnapshot,
399
+ ).toHaveBeenCalled();
400
+ });
401
+
402
+ it('cleanup unsubscribes from all events', () => {
403
+ const { session, cleanup } = setup();
404
+ cleanup();
405
+ expect((session as unknown as { off: ReturnType<typeof vi.fn> }).off).toHaveBeenCalledTimes(11);
406
+ });
407
+ });
@@ -0,0 +1,53 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { createWsTransport } from '../ws-transport.js';
3
+ import type { IInteractiveSession } from '@robota-sdk/agent-framework';
4
+
5
+ function createMockSession(): IInteractiveSession {
6
+ return {
7
+ submit: vi.fn(),
8
+ abort: vi.fn(),
9
+ cancelQueue: vi.fn(),
10
+ getMessages: vi.fn().mockReturnValue([]),
11
+ getContextState: vi
12
+ .fn()
13
+ .mockReturnValue({ usedPercentage: 0, usedTokens: 0, maxTokens: 200000 }),
14
+ isExecuting: vi.fn().mockReturnValue(false),
15
+ getPendingPrompt: vi.fn().mockReturnValue(null),
16
+ executeCommand: vi.fn().mockResolvedValue({ message: 'ok', success: true }),
17
+ listCommands: vi.fn().mockReturnValue([]),
18
+ on: vi.fn(),
19
+ off: vi.fn(),
20
+ } as unknown as IInteractiveSession;
21
+ }
22
+
23
+ describe('createWsTransport', () => {
24
+ it('returns an adapter with name "ws"', () => {
25
+ const transport = createWsTransport({ send: vi.fn() });
26
+ expect(transport.name).toBe('ws');
27
+ });
28
+
29
+ it('throws if start() is called without attach()', async () => {
30
+ const transport = createWsTransport({ send: vi.fn() });
31
+ await expect(transport.start()).rejects.toThrow('No session attached');
32
+ });
33
+
34
+ it('onMessage is null before start()', () => {
35
+ const transport = createWsTransport({ send: vi.fn() });
36
+ expect(transport.onMessage).toBeNull();
37
+ });
38
+
39
+ it('provides onMessage after attach + start', async () => {
40
+ const transport = createWsTransport({ send: vi.fn() });
41
+ transport.attach(createMockSession() as never);
42
+ await transport.start();
43
+ expect(typeof transport.onMessage).toBe('function');
44
+ });
45
+
46
+ it('clears onMessage after stop()', async () => {
47
+ const transport = createWsTransport({ send: vi.fn() });
48
+ transport.attach(createMockSession() as never);
49
+ await transport.start();
50
+ await transport.stop();
51
+ expect(transport.onMessage).toBeNull();
52
+ });
53
+ });
@@ -0,0 +1,13 @@
1
+ export { createWsHandler } from './ws-handler.js';
2
+ export type { IWsHandlerOptions } from './ws-handler.js';
3
+ export type { TClientMessage, TServerMessage } from './ws-protocol.js';
4
+ export { createWsTransport } from './ws-transport.js';
5
+ export type { IWsTransportOptions } from './ws-transport.js';
6
+ export { WsTransport } from './ws-transport-configurable.js';
7
+ export type { IWsTransportConfig } from './ws-transport-configurable.js';
8
+ export type {
9
+ IExecutionWorkspaceSnapshot,
10
+ IExecutionWorkspaceEntry,
11
+ TExecutionWorkspaceStatus,
12
+ TExecutionAttention,
13
+ } from '@robota-sdk/agent-framework';