@robota-sdk/agent-transport 3.0.0-beta.74 → 3.0.0-beta.76

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 (197) hide show
  1. package/README.md +10 -10
  2. package/dist/node/headless/index.cjs +1 -1
  3. package/dist/node/headless/index.d.ts +1 -1
  4. package/dist/node/headless/index.js +1 -1
  5. package/dist/node/headless-OnpVk4-k.cjs +15 -0
  6. package/dist/node/{headless-D02zUEGh.js → headless-mRYilLfC.js} +2 -2
  7. package/dist/node/{headless-D02zUEGh.js.map → headless-mRYilLfC.js.map} +1 -1
  8. package/dist/node/{index-DE3-dHqw.d.ts → index-CYl7ksS6.d.ts} +12 -2
  9. package/dist/node/{index-DE3-dHqw.d.ts.map → index-CYl7ksS6.d.ts.map} +1 -1
  10. package/dist/node/{index-WKTgvhlg.d.ts → index-E8Gx4-lc.d.ts} +12 -2
  11. package/dist/node/{index-WKTgvhlg.d.ts.map → index-E8Gx4-lc.d.ts.map} +1 -1
  12. package/dist/node/index.cjs +1 -1
  13. package/dist/node/index.d.ts +2 -7
  14. package/dist/node/index.d.ts.map +1 -1
  15. package/dist/node/index.js +1 -1
  16. package/dist/node/index.js.map +1 -1
  17. package/package.json +7 -75
  18. package/src/headless/HeadlessInteractionChannel.ts +21 -1
  19. package/src/index.ts +1 -5
  20. package/src/transport-registry.ts +0 -9
  21. package/dist/node/headless-BeHAOlIM.cjs +0 -15
  22. package/dist/node/http/index.cjs +0 -1
  23. package/dist/node/http/index.d.ts +0 -2
  24. package/dist/node/http/index.js +0 -1
  25. package/dist/node/http-2Jiuflc1.js +0 -2
  26. package/dist/node/http-2Jiuflc1.js.map +0 -1
  27. package/dist/node/http-CBAvefLw.cjs +0 -1
  28. package/dist/node/index-BQLN_Lc9.d.ts +0 -78
  29. package/dist/node/index-BQLN_Lc9.d.ts.map +0 -1
  30. package/dist/node/index-BnAGE-u9.d.ts +0 -33
  31. package/dist/node/index-BnAGE-u9.d.ts.map +0 -1
  32. package/dist/node/index-BrQ4gGw0.d.ts +0 -213
  33. package/dist/node/index-BrQ4gGw0.d.ts.map +0 -1
  34. package/dist/node/index-CoeBF21y.d.ts +0 -213
  35. package/dist/node/index-CoeBF21y.d.ts.map +0 -1
  36. package/dist/node/index-DHt-2VQ-.d.ts +0 -46
  37. package/dist/node/index-DHt-2VQ-.d.ts.map +0 -1
  38. package/dist/node/index-DMwKN5Le.d.ts +0 -33
  39. package/dist/node/index-DMwKN5Le.d.ts.map +0 -1
  40. package/dist/node/index-IvYaYY6v.d.ts +0 -78
  41. package/dist/node/index-IvYaYY6v.d.ts.map +0 -1
  42. package/dist/node/index-c0M42fsA.d.ts +0 -46
  43. package/dist/node/index-c0M42fsA.d.ts.map +0 -1
  44. package/dist/node/mcp/index.cjs +0 -1
  45. package/dist/node/mcp/index.d.ts +0 -2
  46. package/dist/node/mcp/index.js +0 -1
  47. package/dist/node/mcp-BOglBJNy.cjs +0 -1
  48. package/dist/node/mcp-D3BBVK7C.js +0 -2
  49. package/dist/node/mcp-D3BBVK7C.js.map +0 -1
  50. package/dist/node/rolldown-runtime-CMqjfN_6.cjs +0 -1
  51. package/dist/node/tui/index.cjs +0 -1
  52. package/dist/node/tui/index.d.ts +0 -2
  53. package/dist/node/tui/index.js +0 -1
  54. package/dist/node/tui-Btb1q88j.js +0 -25
  55. package/dist/node/tui-Btb1q88j.js.map +0 -1
  56. package/dist/node/tui-SbUT7Zlt.cjs +0 -24
  57. package/dist/node/ws/index.cjs +0 -1
  58. package/dist/node/ws/index.d.ts +0 -2
  59. package/dist/node/ws/index.js +0 -1
  60. package/dist/node/ws-Dc2RUwVs.js +0 -2
  61. package/dist/node/ws-Dc2RUwVs.js.map +0 -1
  62. package/dist/node/ws-QNMQn5kg.cjs +0 -1
  63. package/src/http/__tests__/http-transport.test.ts +0 -55
  64. package/src/http/__tests__/routes.test.ts +0 -168
  65. package/src/http/http-transport.ts +0 -41
  66. package/src/http/index.ts +0 -4
  67. package/src/http/routes.ts +0 -152
  68. package/src/mcp/__tests__/mcp-server.test.ts +0 -66
  69. package/src/mcp/__tests__/mcp-transport.test.ts +0 -46
  70. package/src/mcp/index.ts +0 -4
  71. package/src/mcp/mcp-server.ts +0 -163
  72. package/src/mcp/mcp-transport.ts +0 -48
  73. package/src/tui/App.tsx +0 -488
  74. package/src/tui/BackgroundTaskPanel.tsx +0 -36
  75. package/src/tui/CjkTextInput.tsx +0 -199
  76. package/src/tui/ConfirmPrompt.tsx +0 -70
  77. package/src/tui/ContextWarningBanner.tsx +0 -34
  78. package/src/tui/ExecutionWorkspaceDetailPane.tsx +0 -64
  79. package/src/tui/ExecutionWorkspaceSwitcher.tsx +0 -187
  80. package/src/tui/InputArea.tsx +0 -310
  81. package/src/tui/InteractivePrompt.tsx +0 -59
  82. package/src/tui/ListPicker.tsx +0 -95
  83. package/src/tui/MenuSelect.tsx +0 -104
  84. package/src/tui/MessageList.tsx +0 -284
  85. package/src/tui/PermissionPrompt.tsx +0 -86
  86. package/src/tui/PluginTUI.tsx +0 -258
  87. package/src/tui/SessionPicker.tsx +0 -68
  88. package/src/tui/SessionStatusBar.tsx +0 -70
  89. package/src/tui/SlashAutocomplete.tsx +0 -110
  90. package/src/tui/StatusBar.tsx +0 -209
  91. package/src/tui/StreamingIndicator.tsx +0 -93
  92. package/src/tui/TextPrompt.tsx +0 -81
  93. package/src/tui/ToolCommandOutput.tsx +0 -39
  94. package/src/tui/ToolDiffBlock.tsx +0 -32
  95. package/src/tui/TransportTUI.tsx +0 -117
  96. package/src/tui/TuiInteractionChannel.ts +0 -483
  97. package/src/tui/UpdateNotice.tsx +0 -14
  98. package/src/tui/UsageSummaryEntry.tsx +0 -39
  99. package/src/tui/WaveText.tsx +0 -44
  100. package/src/tui/__tests__/InteractivePrompt.test.tsx +0 -82
  101. package/src/tui/__tests__/ListPicker.test.tsx +0 -159
  102. package/src/tui/__tests__/MenuSelect.test.tsx +0 -103
  103. package/src/tui/__tests__/PluginTUI.test.tsx +0 -167
  104. package/src/tui/__tests__/SlashAutocomplete.test.tsx +0 -140
  105. package/src/tui/__tests__/TextPrompt.test.tsx +0 -98
  106. package/src/tui/__tests__/TuiInteractionChannel.display-contract.test.ts +0 -239
  107. package/src/tui/__tests__/TuiInteractionChannel.lifecycle.test.ts +0 -297
  108. package/src/tui/__tests__/TuiInteractionChannel.requestAction.test.ts +0 -124
  109. package/src/tui/__tests__/UpdateNotice.test.tsx +0 -15
  110. package/src/tui/__tests__/abort-after-permission.test.tsx +0 -169
  111. package/src/tui/__tests__/abort-streaming-e2e.test.tsx +0 -183
  112. package/src/tui/__tests__/background-task-panel.test.tsx +0 -53
  113. package/src/tui/__tests__/background-task-row-format.test.ts +0 -59
  114. package/src/tui/__tests__/channel-factory-integration.test.ts +0 -138
  115. package/src/tui/__tests__/cjk-text-input-flow.test.ts +0 -109
  116. package/src/tui/__tests__/cjk-text-input.test.ts +0 -191
  117. package/src/tui/__tests__/command-effect-handler.test.ts +0 -127
  118. package/src/tui/__tests__/command-output-summary.test.ts +0 -95
  119. package/src/tui/__tests__/compact-event-bridge.test.ts +0 -20
  120. package/src/tui/__tests__/confirm-permission-flow.test.ts +0 -130
  121. package/src/tui/__tests__/confirm-prompt.test.tsx +0 -87
  122. package/src/tui/__tests__/execution-workspace-switcher.test.tsx +0 -110
  123. package/src/tui/__tests__/execution-workspace-view-model.test.ts +0 -93
  124. package/src/tui/__tests__/fixtures/provider-setup-prompt-driver.tsx +0 -125
  125. package/src/tui/__tests__/input-area-flow.test.ts +0 -164
  126. package/src/tui/__tests__/message-list-rendering.test.tsx +0 -353
  127. package/src/tui/__tests__/prompt-queue.test.tsx +0 -255
  128. package/src/tui/__tests__/provider-setup-pty-e2e.test.ts +0 -233
  129. package/src/tui/__tests__/pty/pty-driver.ts +0 -135
  130. package/src/tui/__tests__/pty/tui-pty.ptytest.ts +0 -61
  131. package/src/tui/__tests__/render-channel-options.test.ts +0 -32
  132. package/src/tui/__tests__/render-markdown.test.ts +0 -72
  133. package/src/tui/__tests__/selection-flow.test.ts +0 -61
  134. package/src/tui/__tests__/session-init-poller.test.ts +0 -102
  135. package/src/tui/__tests__/session-naming.test.ts +0 -64
  136. package/src/tui/__tests__/session-switch-channel.test.tsx +0 -307
  137. package/src/tui/__tests__/slash-routing-effects.test.ts +0 -228
  138. package/src/tui/__tests__/status-activity.test.ts +0 -71
  139. package/src/tui/__tests__/status-bar.test.tsx +0 -158
  140. package/src/tui/__tests__/streaming-indicator.test.tsx +0 -137
  141. package/src/tui/__tests__/text-prompt-flow.test.ts +0 -77
  142. package/src/tui/__tests__/tui-channel-init-failure.test.ts +0 -57
  143. package/src/tui/__tests__/tui-state-manager.test.ts +0 -401
  144. package/src/tui/background-task-row-format.ts +0 -53
  145. package/src/tui/command-interaction.ts +0 -9
  146. package/src/tui/command-output-summary.ts +0 -122
  147. package/src/tui/create-default-tui-cli-adapter.ts +0 -41
  148. package/src/tui/execution-workspace-view-model.ts +0 -123
  149. package/src/tui/flows/cjk-text-input-flow.ts +0 -285
  150. package/src/tui/flows/confirm-prompt-flow.ts +0 -45
  151. package/src/tui/flows/input-area-flow.ts +0 -189
  152. package/src/tui/flows/permission-prompt-flow.ts +0 -85
  153. package/src/tui/flows/selection-flow.ts +0 -126
  154. package/src/tui/flows/session-init-poller.ts +0 -77
  155. package/src/tui/flows/text-prompt-flow.ts +0 -98
  156. package/src/tui/hooks/command-effect-handler.ts +0 -97
  157. package/src/tui/hooks/command-effect-queue.ts +0 -39
  158. package/src/tui/hooks/side-effects-types.ts +0 -35
  159. package/src/tui/hooks/useAutocomplete.ts +0 -87
  160. package/src/tui/hooks/usePluginCallbacks.ts +0 -31
  161. package/src/tui/hooks/usePluginScreenData.ts +0 -85
  162. package/src/tui/hooks/useSideEffects.ts +0 -175
  163. package/src/tui/hooks/useSlashRouting.ts +0 -118
  164. package/src/tui/hooks/useStatusLineSettings.ts +0 -37
  165. package/src/tui/hooks/useTuiChannel.ts +0 -95
  166. package/src/tui/index.ts +0 -14
  167. package/src/tui/interactions/CommandConfirm.tsx +0 -36
  168. package/src/tui/interactions/CommandPicker.tsx +0 -77
  169. package/src/tui/interactions/__tests__/CommandConfirm.test.tsx +0 -124
  170. package/src/tui/interactions/__tests__/CommandPicker.test.tsx +0 -138
  171. package/src/tui/plugin-tui-handlers.ts +0 -163
  172. package/src/tui/render-markdown.ts +0 -130
  173. package/src/tui/render.tsx +0 -117
  174. package/src/tui/session-naming.ts +0 -33
  175. package/src/tui/status-activity.ts +0 -63
  176. package/src/tui/tui-cli-adapter-context.tsx +0 -13
  177. package/src/tui/tui-cli-adapter.ts +0 -25
  178. package/src/tui/tui-state-manager.ts +0 -226
  179. package/src/tui/tui-transport.ts +0 -35
  180. package/src/tui/types.ts +0 -15
  181. package/src/tui/utils/__tests__/edit-diff.test.ts +0 -426
  182. package/src/tui/utils/__tests__/paste-detection.test.ts +0 -116
  183. package/src/tui/utils/__tests__/paste-labels.test.ts +0 -46
  184. package/src/tui/utils/__tests__/tool-call-extractor.test.ts +0 -227
  185. package/src/tui/utils/__tests__/tool-diff-summary.test.ts +0 -104
  186. package/src/tui/utils/edit-diff.ts +0 -153
  187. package/src/tui/utils/paste-labels.ts +0 -9
  188. package/src/tui/utils/tool-call-extractor.ts +0 -92
  189. package/src/tui/utils/tool-diff-summary.ts +0 -75
  190. package/src/ws/__tests__/ws-handler.test.ts +0 -409
  191. package/src/ws/__tests__/ws-transport.test.ts +0 -53
  192. package/src/ws/index.ts +0 -13
  193. package/src/ws/ws-background-messages.ts +0 -170
  194. package/src/ws/ws-handler.ts +0 -280
  195. package/src/ws/ws-protocol.ts +0 -78
  196. package/src/ws/ws-transport-configurable.ts +0 -128
  197. package/src/ws/ws-transport.ts +0 -42
@@ -1,32 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import type { IAIProvider } from '@robota-sdk/agent-core';
3
- import type { ITuiCliAdapter } from '../tui-cli-adapter.js';
4
- import { toChannelOptions } from '../render.js';
5
- import type { IRenderOptions } from '../render.js';
6
-
7
- describe('toChannelOptions', () => {
8
- it('TC-02: threads allowedTools and deniedTools into the channel options', () => {
9
- const renderOptions: IRenderOptions = {
10
- cwd: '/tmp/project',
11
- provider: {} as IAIProvider,
12
- cliAdapter: {} as ITuiCliAdapter,
13
- allowedTools: ['Read'],
14
- deniedTools: ['Bash'],
15
- };
16
- const channelOptions = toChannelOptions(renderOptions, 'session-1');
17
- expect(channelOptions.allowedTools).toEqual(['Read']);
18
- expect(channelOptions.deniedTools).toEqual(['Bash']);
19
- expect(channelOptions.resumeSessionId).toBe('session-1');
20
- expect(channelOptions.cwd).toBe('/tmp/project');
21
- });
22
-
23
- it('leaves tool filters undefined when not provided', () => {
24
- const channelOptions = toChannelOptions({
25
- cwd: '/tmp/project',
26
- provider: {} as IAIProvider,
27
- cliAdapter: {} as ITuiCliAdapter,
28
- });
29
- expect(channelOptions.allowedTools).toBeUndefined();
30
- expect(channelOptions.deniedTools).toBeUndefined();
31
- });
32
- });
@@ -1,72 +0,0 @@
1
- import { describe, expect, it } from 'vitest';
2
- import { renderMarkdown } from '../render-markdown.js';
3
-
4
- const ANSI_LIGHT_RED = '\u001b[38;5;210m';
5
- const ANSI_LIGHT_GREEN = '\u001b[38;5;120m';
6
- const ANSI_DARK_RED_BACKGROUND = '\u001b[48;5;52m';
7
- const ANSI_DARK_GREEN_BACKGROUND = '\u001b[48;5;22m';
8
- const ANSI_RESET = '\u001b[0m';
9
- const CODE_BLOCK_INDENT = ' ';
10
-
11
- describe('renderMarkdown', () => {
12
- it('renders diff fenced code blocks with addition and removal colors', () => {
13
- const output = renderMarkdown(
14
- ['Before', '', '```diff', '- const oldValue = true;', '+ const newValue = true;', '```'].join(
15
- '\n',
16
- ),
17
- { color: true },
18
- );
19
-
20
- expect(output).toContain(`${ANSI_DARK_RED_BACKGROUND}${ANSI_LIGHT_RED}`);
21
- expect(output).toContain(`${CODE_BLOCK_INDENT}- const oldValue = true;`);
22
- expect(output).toContain(`${ANSI_DARK_GREEN_BACKGROUND}${ANSI_LIGHT_GREEN}`);
23
- expect(output).toContain(`${CODE_BLOCK_INDENT}+ const newValue = true;`);
24
- });
25
-
26
- it('pads added and removed diff rows before applying background colors', () => {
27
- const codeBlockWidth = 24;
28
- const removedRow = `${CODE_BLOCK_INDENT}- removed`.padEnd(codeBlockWidth);
29
- const addedRow = `${CODE_BLOCK_INDENT}+ added`.padEnd(codeBlockWidth);
30
- const output = renderMarkdown(['```diff', '- removed', '+ added', '```'].join('\n'), {
31
- color: true,
32
- codeBlockWidth,
33
- });
34
-
35
- expect(output).toContain(
36
- `${ANSI_DARK_RED_BACKGROUND}${ANSI_LIGHT_RED}${removedRow}${ANSI_RESET}`,
37
- );
38
- expect(output).toContain(
39
- `${ANSI_DARK_GREEN_BACKGROUND}${ANSI_LIGHT_GREEN}${addedRow}${ANSI_RESET}`,
40
- );
41
- });
42
-
43
- it('keeps diff fenced code block content readable when color is disabled', () => {
44
- const output = renderMarkdown(
45
- ['```diff', '- removed line', '+ added line', ' unchanged line', '```'].join('\n'),
46
- { color: false },
47
- );
48
-
49
- expect(output).toContain('- removed line');
50
- expect(output).toContain('+ added line');
51
- expect(output).toContain(' unchanged line');
52
- expect(output).not.toContain(ANSI_LIGHT_RED);
53
- expect(output).not.toContain(ANSI_LIGHT_GREEN);
54
- expect(output).not.toContain(ANSI_DARK_RED_BACKGROUND);
55
- expect(output).not.toContain(ANSI_DARK_GREEN_BACKGROUND);
56
- });
57
-
58
- it('keeps regular fenced code blocks as code output', () => {
59
- const output = renderMarkdown(['```ts', 'const value: string = "ok";', '```'].join('\n'), {
60
- color: false,
61
- });
62
-
63
- expect(output).toContain('const value: string = "ok";');
64
- });
65
-
66
- it('keeps inline markdown formatting readable', () => {
67
- const output = renderMarkdown('Use **bold** and `code` here.', { color: false });
68
-
69
- expect(output).toContain('bold');
70
- expect(output).toContain('code');
71
- });
72
- });
@@ -1,61 +0,0 @@
1
- import { describe, expect, it } from 'vitest';
2
- import {
3
- applySelectionInput,
4
- createSelectionFlowState,
5
- getDirectionalSelectionInputAction,
6
- getVerticalSelectionInputAction,
7
- } from '../flows/selection-flow.js';
8
-
9
- describe('selection flow', () => {
10
- it('Given first item selected When previous is applied Then selection stays bounded', () => {
11
- const result = applySelectionInput(createSelectionFlowState(), 'previous', { itemCount: 3 });
12
-
13
- expect(result.state.selectedIndex).toBe(0);
14
- expect(result.effect).toEqual({ type: 'none' });
15
- });
16
-
17
- it('Given last item selected When next is applied Then selection stays bounded', () => {
18
- const state = { selectedIndex: 2, scrollOffset: 0, resolved: false };
19
-
20
- const result = applySelectionInput(state, 'next', { itemCount: 3 });
21
-
22
- expect(result.state.selectedIndex).toBe(2);
23
- });
24
-
25
- it('Given wrapping selection When previous from first is applied Then it wraps to last', () => {
26
- const result = applySelectionInput(createSelectionFlowState(), 'previous', {
27
- itemCount: 3,
28
- wrap: true,
29
- });
30
-
31
- expect(result.state.selectedIndex).toBe(2);
32
- });
33
-
34
- it('Given max visible window When moving below viewport Then scroll offset follows', () => {
35
- const state = applySelectionInput(createSelectionFlowState(), 'next', {
36
- itemCount: 4,
37
- maxVisible: 2,
38
- }).state;
39
-
40
- const result = applySelectionInput(state, 'next', { itemCount: 4, maxVisible: 2 });
41
-
42
- expect(result.state).toMatchObject({ selectedIndex: 2, scrollOffset: 1 });
43
- });
44
-
45
- it('Given selected item When select is applied Then selected index is emitted once', () => {
46
- const state = { selectedIndex: 1, scrollOffset: 0, resolved: false };
47
-
48
- const selected = applySelectionInput(state, 'select', { itemCount: 3 });
49
- const ignored = applySelectionInput(selected.state, 'select', { itemCount: 3 });
50
-
51
- expect(selected.effect).toEqual({ type: 'select', index: 1 });
52
- expect(ignored.effect).toEqual({ type: 'none' });
53
- });
54
-
55
- it('Given raw key info When mapped Then vertical and directional actions are produced', () => {
56
- expect(getVerticalSelectionInputAction({ downArrow: true })).toBe('next');
57
- expect(getVerticalSelectionInputAction({ escape: true })).toBe('cancel');
58
- expect(getDirectionalSelectionInputAction({ leftArrow: true })).toBe('previous');
59
- expect(getDirectionalSelectionInputAction({ rightArrow: true })).toBe('next');
60
- });
61
- });
@@ -1,102 +0,0 @@
1
- import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
2
-
3
- import { createSessionInitPoller } from '../flows/session-init-poller.js';
4
-
5
- describe('createSessionInitPoller', () => {
6
- beforeEach(() => {
7
- vi.useFakeTimers();
8
- });
9
-
10
- afterEach(() => {
11
- vi.useRealTimers();
12
- });
13
-
14
- it('TC-04: calls onReady and stops once the check succeeds', () => {
15
- const onReady = vi.fn();
16
- const onFailure = vi.fn();
17
- let ready = false;
18
- const poller = createSessionInitPoller({
19
- check: () => {
20
- if (!ready) throw new Error('InteractiveSession not initialized. Call submit().');
21
- },
22
- intervalMs: 200,
23
- timeoutMs: 15000,
24
- onReady,
25
- onFailure,
26
- });
27
- poller.start();
28
- vi.advanceTimersByTime(600);
29
- expect(onReady).not.toHaveBeenCalled();
30
- ready = true;
31
- vi.advanceTimersByTime(200);
32
- expect(onReady).toHaveBeenCalledTimes(1);
33
- vi.advanceTimersByTime(2000);
34
- expect(onReady).toHaveBeenCalledTimes(1);
35
- expect(onFailure).not.toHaveBeenCalled();
36
- });
37
-
38
- it('TC-04: benign not-initialized errors poll until timeout, then fail with timeout kind', () => {
39
- const onReady = vi.fn();
40
- const onFailure = vi.fn();
41
- const poller = createSessionInitPoller({
42
- check: () => {
43
- throw new Error('InteractiveSession not initialized. Call submit().');
44
- },
45
- intervalMs: 200,
46
- timeoutMs: 1000,
47
- onReady,
48
- onFailure,
49
- });
50
- poller.start();
51
- vi.advanceTimersByTime(900);
52
- expect(onFailure).not.toHaveBeenCalled();
53
- vi.advanceTimersByTime(400);
54
- expect(onFailure).toHaveBeenCalledTimes(1);
55
- expect(onFailure.mock.calls[0]?.[0]).toMatchObject({ kind: 'timeout' });
56
- expect(onReady).not.toHaveBeenCalled();
57
- vi.advanceTimersByTime(2000);
58
- expect(onFailure).toHaveBeenCalledTimes(1);
59
- });
60
-
61
- it('TC-04: a real error fails immediately with the error attached', () => {
62
- const onFailure = vi.fn();
63
- const poller = createSessionInitPoller({
64
- check: () => {
65
- throw new Error('ENOENT: session store unreadable');
66
- },
67
- intervalMs: 200,
68
- timeoutMs: 15000,
69
- onReady: vi.fn(),
70
- onFailure,
71
- });
72
- poller.start();
73
- vi.advanceTimersByTime(200);
74
- expect(onFailure).toHaveBeenCalledTimes(1);
75
- expect(onFailure.mock.calls[0]?.[0]).toMatchObject({ kind: 'error' });
76
- expect(String((onFailure.mock.calls[0]?.[0] as { error: Error }).error.message)).toContain(
77
- 'ENOENT',
78
- );
79
- vi.advanceTimersByTime(2000);
80
- expect(onFailure).toHaveBeenCalledTimes(1);
81
- });
82
-
83
- it('stop() cancels polling without callbacks', () => {
84
- const onReady = vi.fn();
85
- const onFailure = vi.fn();
86
- const poller = createSessionInitPoller({
87
- check: () => {
88
- throw new Error('InteractiveSession not initialized.');
89
- },
90
- intervalMs: 200,
91
- timeoutMs: 1000,
92
- onReady,
93
- onFailure,
94
- });
95
- poller.start();
96
- vi.advanceTimersByTime(400);
97
- poller.stop();
98
- vi.advanceTimersByTime(5000);
99
- expect(onReady).not.toHaveBeenCalled();
100
- expect(onFailure).not.toHaveBeenCalled();
101
- });
102
- });
@@ -1,64 +0,0 @@
1
- import { describe, it, expect, vi } from 'vitest';
2
- import { generateSessionName } from '../session-naming.js';
3
- import type { IAIProvider } from '@robota-sdk/agent-core';
4
-
5
- function makeProvider(responseContent: string): IAIProvider {
6
- return {
7
- name: 'mock',
8
- version: '1.0.0',
9
- chat: vi.fn().mockResolvedValue({ role: 'assistant', content: responseContent }),
10
- generateResponse: vi.fn(),
11
- supportsTools: () => false,
12
- } as unknown as IAIProvider;
13
- }
14
-
15
- describe('generateSessionName', () => {
16
- it('returns sanitized name from provider response', async () => {
17
- const provider = makeProvider('refactor-auth-middleware');
18
- const name = await generateSessionName(provider, 'Refactor the auth middleware');
19
- expect(name).toBe('refactor-auth-middleware');
20
- });
21
-
22
- it('lowercases and strips special characters', async () => {
23
- const provider = makeProvider('Fix: Database Connection!');
24
- const name = await generateSessionName(provider, 'Fix database connection');
25
- expect(name).toBe('fix-database-connection');
26
- });
27
-
28
- it('collapses multiple spaces to single hyphen', async () => {
29
- const provider = makeProvider('write api docs');
30
- const name = await generateSessionName(provider, 'Write API docs');
31
- expect(name).toBe('write-api-docs');
32
- });
33
-
34
- it('falls back to sanitized first message when response is too short', async () => {
35
- const provider = makeProvider('ok');
36
- const name = await generateSessionName(provider, 'Fix login bug');
37
- expect(name).toBe('fix-login-bug');
38
- });
39
-
40
- it('truncates long names to 60 chars', async () => {
41
- const long = 'a'.repeat(100);
42
- const provider = makeProvider(long);
43
- const name = await generateSessionName(provider, 'something');
44
- expect(name.length).toBeLessThanOrEqual(60);
45
- });
46
-
47
- it('passes maxTokens: 20 to provider', async () => {
48
- const provider = makeProvider('short-name');
49
- await generateSessionName(provider, 'test');
50
- expect(provider.chat).toHaveBeenCalledWith(
51
- expect.any(Array),
52
- expect.objectContaining({ maxTokens: 20 }),
53
- );
54
- });
55
-
56
- it('truncates first message to 200 chars before sending', async () => {
57
- const long = 'x'.repeat(500);
58
- const provider = makeProvider('short-name');
59
- await generateSessionName(provider, long);
60
- const messages = (provider.chat as ReturnType<typeof vi.fn>).mock.calls[0][0];
61
- const userMsg = messages.find((m: { role: string }) => m.role === 'user');
62
- expect(userMsg.content.length).toBeLessThanOrEqual(200);
63
- });
64
- });
@@ -1,307 +0,0 @@
1
- /**
2
- * CLI-B11 TC-01/03/05 + CLI-B12 TC-01/02/04: session-switch channel ownership
3
- * at the App boundary.
4
- *
5
- * The 2026-05-31 context-loss bug lived between render.tsx, App.tsx and
6
- * TuiInteractionChannel — InteractiveSession-level tests stayed green through it.
7
- * These tests render the REAL App with a mocked createChannel factory and drive
8
- * switches through the real SessionPicker, pinning the factory-call contract.
9
- * Since CLI-B12 the factory is the SOLE channel source: App creates the initial
10
- * channel in its useState initializer and replaces it on every switch.
11
- */
12
-
13
- import { mkdtempSync, rmSync } from 'node:fs';
14
- import { tmpdir } from 'node:os';
15
- import { join } from 'node:path';
16
-
17
- import { render } from 'ink-testing-library';
18
- import React from 'react';
19
- import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
20
-
21
- import App from '../App.js';
22
- import { CommandEffectQueue } from '../hooks/command-effect-queue.js';
23
- import { TuiStateManager } from '../tui-state-manager.js';
24
-
25
- import type { ICommandEffectQueue } from '../hooks/command-effect-queue.js';
26
- import type { ITuiCliAdapter } from '../tui-cli-adapter.js';
27
- import type { TuiInteractionChannel } from '../TuiInteractionChannel.js';
28
- import type {
29
- IInteractiveSessionRecord,
30
- IInteractiveSessionStore,
31
- } from '@robota-sdk/agent-interface-transport';
32
-
33
- const TICK_MS = 30;
34
- const FRAME_DEADLINE_MS = 3000;
35
-
36
- function tick(ms = TICK_MS): Promise<void> {
37
- return new Promise((resolve) => setTimeout(resolve, ms));
38
- }
39
-
40
- async function waitForFrame(
41
- lastFrame: () => string | undefined,
42
- predicate: (frame: string) => boolean,
43
- ): Promise<void> {
44
- const deadline = Date.now() + FRAME_DEADLINE_MS;
45
- while (Date.now() < deadline) {
46
- const frame = lastFrame();
47
- if (frame !== undefined && predicate(frame)) return;
48
- await tick(10);
49
- }
50
- throw new Error(`waitForFrame timeout\n--- frame ---\n${lastFrame() ?? '<none>'}`);
51
- }
52
-
53
- interface IFakeChannel {
54
- sessionName: string | undefined;
55
- stateManager: TuiStateManager;
56
- onChange: (() => void) | null;
57
- isShuttingDown: boolean;
58
- permissionRequest: null;
59
- start: ReturnType<typeof vi.fn>;
60
- stop: ReturnType<typeof vi.fn>;
61
- handleInput: ReturnType<typeof vi.fn>;
62
- abort: ReturnType<typeof vi.fn>;
63
- cancelQueue: ReturnType<typeof vi.fn>;
64
- shutdown: ReturnType<typeof vi.fn>;
65
- selectExecutionWorkspaceEntry: ReturnType<typeof vi.fn>;
66
- readExecutionWorkspaceDetail: ReturnType<typeof vi.fn>;
67
- getSession: () => unknown;
68
- getRegistry: () => unknown;
69
- getCommandEffectQueue: () => ICommandEffectQueue;
70
- /** Test handle: the queue backing getCommandEffectQueue. */
71
- effectQueue: CommandEffectQueue;
72
- /** Test handle: which resumeSessionId this channel was created for. */
73
- createdFor: string | undefined;
74
- }
75
-
76
- function createFakeChannel(createdFor: string | undefined): IFakeChannel {
77
- const effectQueue = new CommandEffectQueue();
78
- const fakeSession = {
79
- getName: (): string | undefined => undefined,
80
- getSession: (): never => {
81
- throw new Error('session not initialized (test fake)');
82
- },
83
- getFullHistory: (): never[] => [],
84
- setName: vi.fn(),
85
- shutdown: vi.fn(async () => {}),
86
- sendAgentJob: vi.fn(async () => {}),
87
- };
88
- const fakeRegistry = {
89
- getCommands: (): never[] => [],
90
- getSubcommands: (): never[] => [],
91
- };
92
- return {
93
- sessionName: undefined,
94
- stateManager: new TuiStateManager(),
95
- onChange: null,
96
- isShuttingDown: false,
97
- permissionRequest: null,
98
- start: vi.fn(async () => {}),
99
- stop: vi.fn(async () => {}),
100
- handleInput: vi.fn(async () => {}),
101
- abort: vi.fn(),
102
- cancelQueue: vi.fn(),
103
- shutdown: vi.fn(async () => {}),
104
- selectExecutionWorkspaceEntry: vi.fn(),
105
- readExecutionWorkspaceDetail: vi.fn(async () => ({ lines: [], title: '' })),
106
- getSession: () => fakeSession,
107
- getRegistry: () => fakeRegistry,
108
- getCommandEffectQueue: () => effectQueue,
109
- effectQueue,
110
- createdFor,
111
- };
112
- }
113
-
114
- function asChannel(fake: IFakeChannel): TuiInteractionChannel {
115
- return fake as unknown as TuiInteractionChannel;
116
- }
117
-
118
- function createFakeStore(records: IInteractiveSessionRecord[]): IInteractiveSessionStore {
119
- return {
120
- save: () => undefined,
121
- load: (id) => records.find((r) => r.id === id),
122
- list: () => records,
123
- delete: () => undefined,
124
- };
125
- }
126
-
127
- function sessionRecord(
128
- id: string,
129
- cwd: string,
130
- updatedAt = '2026-06-13T00:00:00.000Z',
131
- ): IInteractiveSessionRecord {
132
- return {
133
- id,
134
- cwd,
135
- createdAt: '2026-06-13T00:00:00.000Z',
136
- updatedAt,
137
- messages: [
138
- { role: 'user', content: `hello from ${id}` },
139
- { role: 'assistant', content: `reply in ${id}` },
140
- ] as IInteractiveSessionRecord['messages'],
141
- };
142
- }
143
-
144
- /** The picker lists sessions newest-first; bumping updatedAt puts a record on top. */
145
- function touch(records: IInteractiveSessionRecord[], id: string, updatedAt: string): void {
146
- const record = records.find((r) => r.id === id);
147
- if (!record) throw new Error(`no record ${id}`);
148
- record.updatedAt = updatedAt;
149
- }
150
-
151
- function createCliAdapter(settingsPath: string): ITuiCliAdapter {
152
- return {
153
- getUserSettingsPath: () => settingsPath,
154
- readSettings: () => ({}),
155
- writeSettings: vi.fn(),
156
- deleteSettings: vi.fn().mockReturnValue(false),
157
- applyStatusLineSettings: vi.fn(),
158
- reloadPluginCommandSource: vi.fn(),
159
- applyActiveModelChange: vi.fn().mockReturnValue({ applied: true }),
160
- getGitBranch: vi.fn().mockReturnValue(undefined),
161
- getProviderDisplayName: vi.fn((type: string) => type),
162
- };
163
- }
164
-
165
- describe('App session-switch channel ownership (CLI-B11)', () => {
166
- let cwd: string;
167
- let created: IFakeChannel[];
168
- let createChannel: ReturnType<typeof vi.fn>;
169
-
170
- beforeEach(() => {
171
- cwd = mkdtempSync(join(tmpdir(), 'robota-b11-'));
172
- created = [];
173
- createChannel = vi.fn((resumeSessionId?: string) => {
174
- const fake = createFakeChannel(resumeSessionId);
175
- created.push(fake);
176
- return asChannel(fake);
177
- });
178
- });
179
-
180
- afterEach(() => {
181
- rmSync(cwd, { recursive: true, force: true });
182
- });
183
-
184
- function renderApp(options?: { sessionIds?: string[] }) {
185
- const ids = options?.sessionIds ?? ['session-aaaaaaaa', 'session-bbbbbbbb'];
186
- const records = ids.map((id) => sessionRecord(id, cwd));
187
- const store = createFakeStore(records);
188
- const instance = render(
189
- <App
190
- cwd={cwd}
191
- createChannel={createChannel}
192
- sessionStore={store}
193
- showSessionPickerOnStart
194
- cliAdapter={createCliAdapter(join(cwd, 'settings.json'))}
195
- />,
196
- );
197
- return { ...instance, records };
198
- }
199
-
200
- it('TC-01 (B11) / TC-01 (B12): the factory is the sole channel source — once at mount, once per switch with the selected sessionId', async () => {
201
- const { stdin, lastFrame } = renderApp();
202
- await tick();
203
- expect(lastFrame()).toContain('Select a session to resume');
204
-
205
- // CLI-B12 TC-01: initial channel from the useState initializer, exactly once.
206
- expect(createChannel).toHaveBeenCalledTimes(1);
207
- expect(createChannel).toHaveBeenNthCalledWith(1, undefined);
208
-
209
- stdin.write('\r'); // select first item (newest first — equal timestamps keep list order)
210
- await tick();
211
-
212
- // CLI-B11 TC-A: the switch asks the factory for exactly one channel with the id.
213
- expect(createChannel).toHaveBeenCalledTimes(2);
214
- expect(createChannel).toHaveBeenNthCalledWith(2, 'session-aaaaaaaa');
215
- });
216
-
217
- it('TC-03 (B11) / TC-02 (B12): the previous channel is stopped before the new one becomes active', async () => {
218
- const { stdin } = renderApp();
219
- await tick();
220
- const initialChannel = created[0]!;
221
- expect(initialChannel.start).toHaveBeenCalled();
222
-
223
- stdin.write('\r');
224
- await tick();
225
-
226
- // Old channel released: stopped by the switch handler and by the unmounting
227
- // AppInner's effect cleanup (stop() is idempotent by contract).
228
- expect(initialChannel.stop).toHaveBeenCalled();
229
- expect(created).toHaveLength(2);
230
- const newChannel = created[1]!;
231
- expect(newChannel.start).toHaveBeenCalled();
232
- expect(newChannel.stop).not.toHaveBeenCalled();
233
-
234
- // CLI-B12 TC-02 ordering: old stop() was invoked BEFORE the factory built
235
- // the replacement channel (stop-before-active contract).
236
- const stopOrder = initialChannel.stop.mock.invocationCallOrder[0]!;
237
- const replacementOrder = createChannel.mock.invocationCallOrder[1]!;
238
- expect(stopOrder).toBeLessThan(replacementOrder);
239
- });
240
-
241
- it('TC-04 (B12): App renders from the factory alone — no channel prop exists', async () => {
242
- // The old no-factory fallback (B11 TC-D) is deleted with CLI-B12: createChannel
243
- // is required and `channel` is no longer a prop (enforced at the type level —
244
- // passing one is a compile error). This pins the runtime half: a render with
245
- // only the factory boots, starts the initial channel, and keeps rendering.
246
- const { lastFrame } = renderApp();
247
- await tick();
248
-
249
- expect(lastFrame()).toBeTruthy();
250
- expect(createChannel).toHaveBeenCalledTimes(1);
251
- expect(created[0]!.start).toHaveBeenCalled();
252
- });
253
-
254
- it('TC-05: consecutive switches A→B→C create one channel per switch and stop each prior channel', async () => {
255
- // Selection always takes the top (newest) entry; arrow-key navigation itself
256
- // is covered by ListPicker.test.tsx. updatedAt ordering decides the target.
257
- const ids = ['aaaaaaaa-1111', 'bbbbbbbb-2222', 'cccccccc-3333'];
258
- const { stdin, lastFrame, records } = renderApp({ sessionIds: ids });
259
- touch(records, 'aaaaaaaa-1111', '2026-06-13T01:00:00.000Z'); // A on top
260
- await waitForFrame(lastFrame, (f) => f.includes('Select a session to resume'));
261
-
262
- // Mount creates the initial channel (factory call 1, undefined).
263
- expect(createChannel).toHaveBeenNthCalledWith(1, undefined);
264
- const channelInitial = created[0]!;
265
-
266
- // Switch 1: pick A (top) from the startup picker.
267
- stdin.write('\r');
268
- await waitForFrame(lastFrame, () => createChannel.mock.calls.length === 2);
269
- expect(createChannel).toHaveBeenNthCalledWith(2, 'aaaaaaaa-1111');
270
- expect(channelInitial.stop).toHaveBeenCalled();
271
- const channelA = created[1]!;
272
-
273
- // Switch 2: reopen the picker via a queued session-picker-requested effect,
274
- // drained by a submit on the active channel (real /resume drain path).
275
- touch(records, 'bbbbbbbb-2222', '2026-06-13T02:00:00.000Z'); // B on top
276
- channelA.effectQueue.enqueueEffects([{ type: 'session-picker-requested' }]);
277
- stdin.write('x');
278
- await tick();
279
- stdin.write('\r'); // submit input → drains queue → picker opens
280
- await waitForFrame(lastFrame, (f) => f.includes('> bbbbbbbb'));
281
- await tick(); // settle: let the reopened picker's useInput subscription attach
282
- stdin.write('\r');
283
- await waitForFrame(lastFrame, () => createChannel.mock.calls.length === 3);
284
- expect(createChannel).toHaveBeenNthCalledWith(3, 'bbbbbbbb-2222');
285
- expect(channelA.stop).toHaveBeenCalled();
286
- const channelB = created[2]!;
287
- expect(channelB.start).toHaveBeenCalled();
288
-
289
- // Switch 3: same drill from B to C.
290
- touch(records, 'cccccccc-3333', '2026-06-13T03:00:00.000Z'); // C on top
291
- channelB.effectQueue.enqueueEffects([{ type: 'session-picker-requested' }]);
292
- stdin.write('x');
293
- await tick();
294
- stdin.write('\r');
295
- await waitForFrame(lastFrame, (f) => f.includes('> cccccccc'));
296
- await tick(); // settle: let the reopened picker's useInput subscription attach
297
- stdin.write('\r');
298
- await waitForFrame(lastFrame, () => createChannel.mock.calls.length === 4);
299
- expect(createChannel).toHaveBeenNthCalledWith(4, 'cccccccc-3333');
300
- expect(channelB.stop).toHaveBeenCalled();
301
-
302
- const channelC = created[3]!;
303
- expect(channelC.start).toHaveBeenCalled();
304
- expect(channelC.stop).not.toHaveBeenCalled();
305
- expect(createChannel).toHaveBeenCalledTimes(4);
306
- });
307
- });